Bluetooth: Handle HCI raw socket transition from unbound to bound
authorMarcel Holtmann <marcel@holtmann.org>
Tue, 30 Aug 2016 03:00:40 +0000 (05:00 +0200)
committerMarcel Holtmann <marcel@holtmann.org>
Mon, 19 Sep 2016 18:19:34 +0000 (20:19 +0200)
In case an unbound HCI raw socket is later on bound, ensure that the
monitor notification messages indicate a close and re-open. None of
the userspace tools use the socket this, but it is actually possible
to use an ioctl on an unbound socket and then later bind it.

Signed-off-by: Marcel Holtmann <marcel@holtmann.org>
Signed-off-by: Johan Hedberg <johan.hedberg@intel.com>
net/bluetooth/hci_sock.c

index c7772436f508186319436e0db60e8916977c02b8..83e9fdb712e522e28e243da6f21afa1c61cd8ea3 100644 (file)
@@ -1049,6 +1049,7 @@ static int hci_sock_bind(struct socket *sock, struct sockaddr *addr,
        struct sockaddr_hci haddr;
        struct sock *sk = sock->sk;
        struct hci_dev *hdev = NULL;
+       struct sk_buff *skb;
        int len, err = 0;
 
        BT_DBG("sock %p sk %p", sock, sk);
@@ -1088,27 +1089,34 @@ static int hci_sock_bind(struct socket *sock, struct sockaddr *addr,
                }
 
                hci_pi(sk)->channel = haddr.hci_channel;
-               hci_pi(sk)->hdev = hdev;
-
-               /* Only send the event to monitor when a new cookie has
-                * been generated. An existing cookie means that an unbound
-                * socket has seen an ioctl and that triggered the cookie
-                * generation and sending of the monitor event.
-                */
-               if (hci_sock_gen_cookie(sk)) {
-                       struct sk_buff *skb;
-
-                       if (capable(CAP_NET_ADMIN))
-                               hci_sock_set_flag(sk, HCI_SOCK_TRUSTED);
 
-                       /* Send event to monitor */
-                       skb = create_monitor_ctrl_open(sk);
+               if (!hci_sock_gen_cookie(sk)) {
+                       /* In the case when a cookie has already been assigned,
+                        * then there has been already an ioctl issued against
+                        * an unbound socket and with that triggerd an open
+                        * notification. Send a close notification first to
+                        * allow the state transition to bounded.
+                        */
+                       skb = create_monitor_ctrl_close(sk);
                        if (skb) {
                                hci_send_to_channel(HCI_CHANNEL_MONITOR, skb,
                                                    HCI_SOCK_TRUSTED, NULL);
                                kfree_skb(skb);
                        }
                }
+
+               if (capable(CAP_NET_ADMIN))
+                       hci_sock_set_flag(sk, HCI_SOCK_TRUSTED);
+
+               hci_pi(sk)->hdev = hdev;
+
+               /* Send event to monitor */
+               skb = create_monitor_ctrl_open(sk);
+               if (skb) {
+                       hci_send_to_channel(HCI_CHANNEL_MONITOR, skb,
+                                           HCI_SOCK_TRUSTED, NULL);
+                       kfree_skb(skb);
+               }
                break;
 
        case HCI_CHANNEL_USER:
@@ -1251,9 +1259,20 @@ static int hci_sock_bind(struct socket *sock, struct sockaddr *addr,
                 * are changes to settings, class of device, name etc.
                 */
                if (hci_pi(sk)->channel == HCI_CHANNEL_CONTROL) {
-                       struct sk_buff *skb;
-
-                       hci_sock_gen_cookie(sk);
+                       if (!hci_sock_gen_cookie(sk)) {
+                               /* In the case when a cookie has already been
+                                * assigned, this socket will transtion from
+                                * a raw socket into a control socket. To
+                                * allow for a clean transtion, send the
+                                * close notification first.
+                                */
+                               skb = create_monitor_ctrl_close(sk);
+                               if (skb) {
+                                       hci_send_to_channel(HCI_CHANNEL_MONITOR, skb,
+                                                           HCI_SOCK_TRUSTED, NULL);
+                                       kfree_skb(skb);
+                               }
+                       }
 
                        /* Send event to monitor */
                        skb = create_monitor_ctrl_open(sk);