]> git.karo-electronics.de Git - mv-sheeva.git/blobdiff - net/bluetooth/hci_core.c
Merge branch 'for-linus' of git://git.infradead.org/users/eparis/notify
[mv-sheeva.git] / net / bluetooth / hci_core.c
index 2f768de8701178091008cca90346647790fe6706..c52f091ee6de2c4b1e81f68363086e2af4a51660 100644 (file)
@@ -562,6 +562,7 @@ static int hci_dev_do_close(struct hci_dev *hdev)
        hci_dev_lock_bh(hdev);
        inquiry_cache_flush(hdev);
        hci_conn_hash_flush(hdev);
+       hci_blacklist_clear(hdev);
        hci_dev_unlock_bh(hdev);
 
        hci_notify(hdev, HCI_DEV_DOWN);
@@ -913,7 +914,7 @@ int hci_register_dev(struct hci_dev *hdev)
        skb_queue_head_init(&hdev->cmd_q);
        skb_queue_head_init(&hdev->raw_q);
 
-       for (i = 0; i < 3; i++)
+       for (i = 0; i < NUM_REASSEMBLY; i++)
                hdev->reassembly[i] = NULL;
 
        init_waitqueue_head(&hdev->req_wait_q);
@@ -923,6 +924,8 @@ int hci_register_dev(struct hci_dev *hdev)
 
        hci_conn_hash_init(hdev);
 
+       INIT_LIST_HEAD(&hdev->blacklist);
+
        memset(&hdev->stat, 0, sizeof(struct hci_dev_stats));
 
        atomic_set(&hdev->promisc, 0);
@@ -970,7 +973,7 @@ int hci_unregister_dev(struct hci_dev *hdev)
 
        hci_dev_do_close(hdev);
 
-       for (i = 0; i < 3; i++)
+       for (i = 0; i < NUM_REASSEMBLY; i++)
                kfree_skb(hdev->reassembly[i]);
 
        hci_notify(hdev, HCI_DEV_UNREG);
@@ -1030,89 +1033,170 @@ int hci_recv_frame(struct sk_buff *skb)
 }
 EXPORT_SYMBOL(hci_recv_frame);
 
-/* Receive packet type fragment */
-#define __reassembly(hdev, type)  ((hdev)->reassembly[(type) - 2])
-
-int hci_recv_fragment(struct hci_dev *hdev, int type, void *data, int count)
+static int hci_reassembly(struct hci_dev *hdev, int type, void *data,
+                         int count, __u8 index, gfp_t gfp_mask)
 {
-       if (type < HCI_ACLDATA_PKT || type > HCI_EVENT_PKT)
+       int len = 0;
+       int hlen = 0;
+       int remain = count;
+       struct sk_buff *skb;
+       struct bt_skb_cb *scb;
+
+       if ((type < HCI_ACLDATA_PKT || type > HCI_EVENT_PKT) ||
+                               index >= NUM_REASSEMBLY)
                return -EILSEQ;
 
+       skb = hdev->reassembly[index];
+
+       if (!skb) {
+               switch (type) {
+               case HCI_ACLDATA_PKT:
+                       len = HCI_MAX_FRAME_SIZE;
+                       hlen = HCI_ACL_HDR_SIZE;
+                       break;
+               case HCI_EVENT_PKT:
+                       len = HCI_MAX_EVENT_SIZE;
+                       hlen = HCI_EVENT_HDR_SIZE;
+                       break;
+               case HCI_SCODATA_PKT:
+                       len = HCI_MAX_SCO_SIZE;
+                       hlen = HCI_SCO_HDR_SIZE;
+                       break;
+               }
+
+               skb = bt_skb_alloc(len, gfp_mask);
+               if (!skb)
+                       return -ENOMEM;
+
+               scb = (void *) skb->cb;
+               scb->expect = hlen;
+               scb->pkt_type = type;
+
+               skb->dev = (void *) hdev;
+               hdev->reassembly[index] = skb;
+       }
+
        while (count) {
-               struct sk_buff *skb = __reassembly(hdev, type);
-               struct { int expect; } *scb;
-               int len = 0;
+               scb = (void *) skb->cb;
+               len = min(scb->expect, (__u16)count);
 
-               if (!skb) {
-                       /* Start of the frame */
+               memcpy(skb_put(skb, len), data, len);
 
-                       switch (type) {
-                       case HCI_EVENT_PKT:
-                               if (count >= HCI_EVENT_HDR_SIZE) {
-                                       struct hci_event_hdr *h = data;
-                                       len = HCI_EVENT_HDR_SIZE + h->plen;
-                               } else
-                                       return -EILSEQ;
-                               break;
+               count -= len;
+               data += len;
+               scb->expect -= len;
+               remain = count;
 
-                       case HCI_ACLDATA_PKT:
-                               if (count >= HCI_ACL_HDR_SIZE) {
-                                       struct hci_acl_hdr *h = data;
-                                       len = HCI_ACL_HDR_SIZE + __le16_to_cpu(h->dlen);
-                               } else
-                                       return -EILSEQ;
-                               break;
+               switch (type) {
+               case HCI_EVENT_PKT:
+                       if (skb->len == HCI_EVENT_HDR_SIZE) {
+                               struct hci_event_hdr *h = hci_event_hdr(skb);
+                               scb->expect = h->plen;
+
+                               if (skb_tailroom(skb) < scb->expect) {
+                                       kfree_skb(skb);
+                                       hdev->reassembly[index] = NULL;
+                                       return -ENOMEM;
+                               }
+                       }
+                       break;
 
-                       case HCI_SCODATA_PKT:
-                               if (count >= HCI_SCO_HDR_SIZE) {
-                                       struct hci_sco_hdr *h = data;
-                                       len = HCI_SCO_HDR_SIZE + h->dlen;
-                               } else
-                                       return -EILSEQ;
-                               break;
+               case HCI_ACLDATA_PKT:
+                       if (skb->len  == HCI_ACL_HDR_SIZE) {
+                               struct hci_acl_hdr *h = hci_acl_hdr(skb);
+                               scb->expect = __le16_to_cpu(h->dlen);
+
+                               if (skb_tailroom(skb) < scb->expect) {
+                                       kfree_skb(skb);
+                                       hdev->reassembly[index] = NULL;
+                                       return -ENOMEM;
+                               }
                        }
+                       break;
 
-                       skb = bt_skb_alloc(len, GFP_ATOMIC);
-                       if (!skb) {
-                               BT_ERR("%s no memory for packet", hdev->name);
-                               return -ENOMEM;
+               case HCI_SCODATA_PKT:
+                       if (skb->len == HCI_SCO_HDR_SIZE) {
+                               struct hci_sco_hdr *h = hci_sco_hdr(skb);
+                               scb->expect = h->dlen;
+
+                               if (skb_tailroom(skb) < scb->expect) {
+                                       kfree_skb(skb);
+                                       hdev->reassembly[index] = NULL;
+                                       return -ENOMEM;
+                               }
                        }
+                       break;
+               }
+
+               if (scb->expect == 0) {
+                       /* Complete frame */
 
-                       skb->dev = (void *) hdev;
                        bt_cb(skb)->pkt_type = type;
+                       hci_recv_frame(skb);
 
-                       __reassembly(hdev, type) = skb;
+                       hdev->reassembly[index] = NULL;
+                       return remain;
+               }
+       }
 
-                       scb = (void *) skb->cb;
-                       scb->expect = len;
-               } else {
-                       /* Continuation */
+       return remain;
+}
 
-                       scb = (void *) skb->cb;
-                       len = scb->expect;
-               }
+int hci_recv_fragment(struct hci_dev *hdev, int type, void *data, int count)
+{
+       int rem = 0;
 
-               len = min(len, count);
+       if (type < HCI_ACLDATA_PKT || type > HCI_EVENT_PKT)
+               return -EILSEQ;
 
-               memcpy(skb_put(skb, len), data, len);
+       while (count) {
+               rem = hci_reassembly(hdev, type, data, count,
+                                               type - 1, GFP_ATOMIC);
+               if (rem < 0)
+                       return rem;
 
-               scb->expect -= len;
+               data += (count - rem);
+               count = rem;
+       };
 
-               if (scb->expect == 0) {
-                       /* Complete frame */
+       return rem;
+}
+EXPORT_SYMBOL(hci_recv_fragment);
 
-                       __reassembly(hdev, type) = NULL;
+#define STREAM_REASSEMBLY 0
 
-                       bt_cb(skb)->pkt_type = type;
-                       hci_recv_frame(skb);
-               }
+int hci_recv_stream_fragment(struct hci_dev *hdev, void *data, int count)
+{
+       int type;
+       int rem = 0;
 
-               count -= len; data += len;
-       }
+       while (count) {
+               struct sk_buff *skb = hdev->reassembly[STREAM_REASSEMBLY];
 
-       return 0;
+               if (!skb) {
+                       struct { char type; } *pkt;
+
+                       /* Start of the frame */
+                       pkt = data;
+                       type = pkt->type;
+
+                       data++;
+                       count--;
+               } else
+                       type = bt_cb(skb)->pkt_type;
+
+               rem = hci_reassembly(hdev, type, data,
+                                       count, STREAM_REASSEMBLY, GFP_ATOMIC);
+               if (rem < 0)
+                       return rem;
+
+               data += (count - rem);
+               count = rem;
+       };
+
+       return rem;
 }
-EXPORT_SYMBOL(hci_recv_fragment);
+EXPORT_SYMBOL(hci_recv_stream_fragment);
 
 /* ---- Interface to upper protocols ---- */