11da177e4SLinus Torvalds /* 21da177e4SLinus Torvalds BlueZ - Bluetooth protocol stack for Linux 31da177e4SLinus Torvalds Copyright (C) 2000-2001 Qualcomm Incorporated 41da177e4SLinus Torvalds 51da177e4SLinus Torvalds Written 2000,2001 by Maxim Krasnyansky <maxk@qualcomm.com> 61da177e4SLinus Torvalds 71da177e4SLinus Torvalds This program is free software; you can redistribute it and/or modify 81da177e4SLinus Torvalds it under the terms of the GNU General Public License version 2 as 91da177e4SLinus Torvalds published by the Free Software Foundation; 101da177e4SLinus Torvalds 111da177e4SLinus Torvalds THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS 121da177e4SLinus Torvalds OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 131da177e4SLinus Torvalds FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT OF THIRD PARTY RIGHTS. 141da177e4SLinus Torvalds IN NO EVENT SHALL THE COPYRIGHT HOLDER(S) AND AUTHOR(S) BE LIABLE FOR ANY 151da177e4SLinus Torvalds CLAIM, OR ANY SPECIAL INDIRECT OR CONSEQUENTIAL DAMAGES, OR ANY DAMAGES 161da177e4SLinus Torvalds WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN 171da177e4SLinus Torvalds ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF 181da177e4SLinus Torvalds OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. 191da177e4SLinus Torvalds 201da177e4SLinus Torvalds ALL LIABILITY, INCLUDING LIABILITY FOR INFRINGEMENT OF ANY PATENTS, 211da177e4SLinus Torvalds COPYRIGHTS, TRADEMARKS OR OTHER RIGHTS, RELATING TO USE OF THIS 221da177e4SLinus Torvalds SOFTWARE IS DISCLAIMED. 231da177e4SLinus Torvalds */ 241da177e4SLinus Torvalds 251da177e4SLinus Torvalds /* Bluetooth HCI sockets. */ 261da177e4SLinus Torvalds 278c520a59SGustavo Padovan #include <linux/export.h> 281da177e4SLinus Torvalds #include <asm/unaligned.h> 291da177e4SLinus Torvalds 301da177e4SLinus Torvalds #include <net/bluetooth/bluetooth.h> 311da177e4SLinus Torvalds #include <net/bluetooth/hci_core.h> 32cd82e61cSMarcel Holtmann #include <net/bluetooth/hci_mon.h> 33fa4335d7SJohan Hedberg #include <net/bluetooth/mgmt.h> 34fa4335d7SJohan Hedberg 35fa4335d7SJohan Hedberg #include "mgmt_util.h" 361da177e4SLinus Torvalds 37801c1e8dSJohan Hedberg static LIST_HEAD(mgmt_chan_list); 38801c1e8dSJohan Hedberg static DEFINE_MUTEX(mgmt_chan_list_lock); 39801c1e8dSJohan Hedberg 40cd82e61cSMarcel Holtmann static atomic_t monitor_promisc = ATOMIC_INIT(0); 41cd82e61cSMarcel Holtmann 421da177e4SLinus Torvalds /* ----- HCI socket interface ----- */ 431da177e4SLinus Torvalds 44863def58SMarcel Holtmann /* Socket info */ 45863def58SMarcel Holtmann #define hci_pi(sk) ((struct hci_pinfo *) sk) 46863def58SMarcel Holtmann 47863def58SMarcel Holtmann struct hci_pinfo { 48863def58SMarcel Holtmann struct bt_sock bt; 49863def58SMarcel Holtmann struct hci_dev *hdev; 50863def58SMarcel Holtmann struct hci_filter filter; 51863def58SMarcel Holtmann __u32 cmsg_mask; 52863def58SMarcel Holtmann unsigned short channel; 536befc644SMarcel Holtmann unsigned long flags; 54863def58SMarcel Holtmann }; 55863def58SMarcel Holtmann 566befc644SMarcel Holtmann void hci_sock_set_flag(struct sock *sk, int nr) 576befc644SMarcel Holtmann { 586befc644SMarcel Holtmann set_bit(nr, &hci_pi(sk)->flags); 596befc644SMarcel Holtmann } 606befc644SMarcel Holtmann 616befc644SMarcel Holtmann void hci_sock_clear_flag(struct sock *sk, int nr) 626befc644SMarcel Holtmann { 636befc644SMarcel Holtmann clear_bit(nr, &hci_pi(sk)->flags); 646befc644SMarcel Holtmann } 656befc644SMarcel Holtmann 66c85be545SMarcel Holtmann int hci_sock_test_flag(struct sock *sk, int nr) 67c85be545SMarcel Holtmann { 68c85be545SMarcel Holtmann return test_bit(nr, &hci_pi(sk)->flags); 69c85be545SMarcel Holtmann } 70c85be545SMarcel Holtmann 71d0f172b1SJohan Hedberg unsigned short hci_sock_get_channel(struct sock *sk) 72d0f172b1SJohan Hedberg { 73d0f172b1SJohan Hedberg return hci_pi(sk)->channel; 74d0f172b1SJohan Hedberg } 75d0f172b1SJohan Hedberg 769391976aSJiri Slaby static inline int hci_test_bit(int nr, const void *addr) 771da177e4SLinus Torvalds { 789391976aSJiri Slaby return *((const __u32 *) addr + (nr >> 5)) & ((__u32) 1 << (nr & 31)); 791da177e4SLinus Torvalds } 801da177e4SLinus Torvalds 811da177e4SLinus Torvalds /* Security filter */ 823ad254f7SMarcel Holtmann #define HCI_SFLT_MAX_OGF 5 833ad254f7SMarcel Holtmann 843ad254f7SMarcel Holtmann struct hci_sec_filter { 853ad254f7SMarcel Holtmann __u32 type_mask; 863ad254f7SMarcel Holtmann __u32 event_mask[2]; 873ad254f7SMarcel Holtmann __u32 ocf_mask[HCI_SFLT_MAX_OGF + 1][4]; 883ad254f7SMarcel Holtmann }; 893ad254f7SMarcel Holtmann 907e67c112SMarcel Holtmann static const struct hci_sec_filter hci_sec_filter = { 911da177e4SLinus Torvalds /* Packet types */ 921da177e4SLinus Torvalds 0x10, 931da177e4SLinus Torvalds /* Events */ 94dd7f5527SMarcel Holtmann { 0x1000d9fe, 0x0000b00c }, 951da177e4SLinus Torvalds /* Commands */ 961da177e4SLinus Torvalds { 971da177e4SLinus Torvalds { 0x0 }, 981da177e4SLinus Torvalds /* OGF_LINK_CTL */ 997c631a67SMarcel Holtmann { 0xbe000006, 0x00000001, 0x00000000, 0x00 }, 1001da177e4SLinus Torvalds /* OGF_LINK_POLICY */ 1017c631a67SMarcel Holtmann { 0x00005200, 0x00000000, 0x00000000, 0x00 }, 1021da177e4SLinus Torvalds /* OGF_HOST_CTL */ 1037c631a67SMarcel Holtmann { 0xaab00200, 0x2b402aaa, 0x05220154, 0x00 }, 1041da177e4SLinus Torvalds /* OGF_INFO_PARAM */ 1057c631a67SMarcel Holtmann { 0x000002be, 0x00000000, 0x00000000, 0x00 }, 1061da177e4SLinus Torvalds /* OGF_STATUS_PARAM */ 1077c631a67SMarcel Holtmann { 0x000000ea, 0x00000000, 0x00000000, 0x00 } 1081da177e4SLinus Torvalds } 1091da177e4SLinus Torvalds }; 1101da177e4SLinus Torvalds 1111da177e4SLinus Torvalds static struct bt_sock_list hci_sk_list = { 112d5fb2962SRobert P. J. Day .lock = __RW_LOCK_UNLOCKED(hci_sk_list.lock) 1131da177e4SLinus Torvalds }; 1141da177e4SLinus Torvalds 115f81fe64fSMarcel Holtmann static bool is_filtered_packet(struct sock *sk, struct sk_buff *skb) 116f81fe64fSMarcel Holtmann { 117f81fe64fSMarcel Holtmann struct hci_filter *flt; 118f81fe64fSMarcel Holtmann int flt_type, flt_event; 119f81fe64fSMarcel Holtmann 120f81fe64fSMarcel Holtmann /* Apply filter */ 121f81fe64fSMarcel Holtmann flt = &hci_pi(sk)->filter; 122f81fe64fSMarcel Holtmann 123f81fe64fSMarcel Holtmann if (bt_cb(skb)->pkt_type == HCI_VENDOR_PKT) 124f81fe64fSMarcel Holtmann flt_type = 0; 125f81fe64fSMarcel Holtmann else 126f81fe64fSMarcel Holtmann flt_type = bt_cb(skb)->pkt_type & HCI_FLT_TYPE_BITS; 127f81fe64fSMarcel Holtmann 128f81fe64fSMarcel Holtmann if (!test_bit(flt_type, &flt->type_mask)) 129f81fe64fSMarcel Holtmann return true; 130f81fe64fSMarcel Holtmann 131f81fe64fSMarcel Holtmann /* Extra filter for event packets only */ 132f81fe64fSMarcel Holtmann if (bt_cb(skb)->pkt_type != HCI_EVENT_PKT) 133f81fe64fSMarcel Holtmann return false; 134f81fe64fSMarcel Holtmann 135f81fe64fSMarcel Holtmann flt_event = (*(__u8 *)skb->data & HCI_FLT_EVENT_BITS); 136f81fe64fSMarcel Holtmann 137f81fe64fSMarcel Holtmann if (!hci_test_bit(flt_event, &flt->event_mask)) 138f81fe64fSMarcel Holtmann return true; 139f81fe64fSMarcel Holtmann 140f81fe64fSMarcel Holtmann /* Check filter only when opcode is set */ 141f81fe64fSMarcel Holtmann if (!flt->opcode) 142f81fe64fSMarcel Holtmann return false; 143f81fe64fSMarcel Holtmann 144f81fe64fSMarcel Holtmann if (flt_event == HCI_EV_CMD_COMPLETE && 145f81fe64fSMarcel Holtmann flt->opcode != get_unaligned((__le16 *)(skb->data + 3))) 146f81fe64fSMarcel Holtmann return true; 147f81fe64fSMarcel Holtmann 148f81fe64fSMarcel Holtmann if (flt_event == HCI_EV_CMD_STATUS && 149f81fe64fSMarcel Holtmann flt->opcode != get_unaligned((__le16 *)(skb->data + 4))) 150f81fe64fSMarcel Holtmann return true; 151f81fe64fSMarcel Holtmann 152f81fe64fSMarcel Holtmann return false; 153f81fe64fSMarcel Holtmann } 154f81fe64fSMarcel Holtmann 1551da177e4SLinus Torvalds /* Send frame to RAW socket */ 156470fe1b5SMarcel Holtmann void hci_send_to_sock(struct hci_dev *hdev, struct sk_buff *skb) 1571da177e4SLinus Torvalds { 1581da177e4SLinus Torvalds struct sock *sk; 159e0edf373SMarcel Holtmann struct sk_buff *skb_copy = NULL; 1601da177e4SLinus Torvalds 1611da177e4SLinus Torvalds BT_DBG("hdev %p len %d", hdev, skb->len); 1621da177e4SLinus Torvalds 1631da177e4SLinus Torvalds read_lock(&hci_sk_list.lock); 164470fe1b5SMarcel Holtmann 165b67bfe0dSSasha Levin sk_for_each(sk, &hci_sk_list.head) { 1661da177e4SLinus Torvalds struct sk_buff *nskb; 1671da177e4SLinus Torvalds 1681da177e4SLinus Torvalds if (sk->sk_state != BT_BOUND || hci_pi(sk)->hdev != hdev) 1691da177e4SLinus Torvalds continue; 1701da177e4SLinus Torvalds 1711da177e4SLinus Torvalds /* Don't send frame to the socket it came from */ 1721da177e4SLinus Torvalds if (skb->sk == sk) 1731da177e4SLinus Torvalds continue; 1741da177e4SLinus Torvalds 17523500189SMarcel Holtmann if (hci_pi(sk)->channel == HCI_CHANNEL_RAW) { 176f81fe64fSMarcel Holtmann if (is_filtered_packet(sk, skb)) 1771da177e4SLinus Torvalds continue; 17823500189SMarcel Holtmann } else if (hci_pi(sk)->channel == HCI_CHANNEL_USER) { 17923500189SMarcel Holtmann if (!bt_cb(skb)->incoming) 18023500189SMarcel Holtmann continue; 18123500189SMarcel Holtmann if (bt_cb(skb)->pkt_type != HCI_EVENT_PKT && 18223500189SMarcel Holtmann bt_cb(skb)->pkt_type != HCI_ACLDATA_PKT && 18323500189SMarcel Holtmann bt_cb(skb)->pkt_type != HCI_SCODATA_PKT) 18423500189SMarcel Holtmann continue; 18523500189SMarcel Holtmann } else { 18623500189SMarcel Holtmann /* Don't send frame to other channel types */ 18723500189SMarcel Holtmann continue; 18823500189SMarcel Holtmann } 1891da177e4SLinus Torvalds 190e0edf373SMarcel Holtmann if (!skb_copy) { 191e0edf373SMarcel Holtmann /* Create a private copy with headroom */ 192bad93e9dSOctavian Purdila skb_copy = __pskb_copy_fclone(skb, 1, GFP_ATOMIC, true); 193e0edf373SMarcel Holtmann if (!skb_copy) 1941da177e4SLinus Torvalds continue; 1951da177e4SLinus Torvalds 1961da177e4SLinus Torvalds /* Put type byte before the data */ 197e0edf373SMarcel Holtmann memcpy(skb_push(skb_copy, 1), &bt_cb(skb)->pkt_type, 1); 198e0edf373SMarcel Holtmann } 199e0edf373SMarcel Holtmann 200e0edf373SMarcel Holtmann nskb = skb_clone(skb_copy, GFP_ATOMIC); 201e0edf373SMarcel Holtmann if (!nskb) 202e0edf373SMarcel Holtmann continue; 2031da177e4SLinus Torvalds 2041da177e4SLinus Torvalds if (sock_queue_rcv_skb(sk, nskb)) 2051da177e4SLinus Torvalds kfree_skb(nskb); 2061da177e4SLinus Torvalds } 207470fe1b5SMarcel Holtmann 208470fe1b5SMarcel Holtmann read_unlock(&hci_sk_list.lock); 209e0edf373SMarcel Holtmann 210e0edf373SMarcel Holtmann kfree_skb(skb_copy); 211470fe1b5SMarcel Holtmann } 212470fe1b5SMarcel Holtmann 2137129069eSJohan Hedberg /* Send frame to sockets with specific channel */ 2147129069eSJohan Hedberg void hci_send_to_channel(unsigned short channel, struct sk_buff *skb, 215c08b1a1dSMarcel Holtmann int flag, struct sock *skip_sk) 216470fe1b5SMarcel Holtmann { 217470fe1b5SMarcel Holtmann struct sock *sk; 218470fe1b5SMarcel Holtmann 2197129069eSJohan Hedberg BT_DBG("channel %u len %d", channel, skb->len); 220470fe1b5SMarcel Holtmann 221470fe1b5SMarcel Holtmann read_lock(&hci_sk_list.lock); 222470fe1b5SMarcel Holtmann 223b67bfe0dSSasha Levin sk_for_each(sk, &hci_sk_list.head) { 224470fe1b5SMarcel Holtmann struct sk_buff *nskb; 225470fe1b5SMarcel Holtmann 226c08b1a1dSMarcel Holtmann /* Ignore socket without the flag set */ 227c85be545SMarcel Holtmann if (!hci_sock_test_flag(sk, flag)) 228c08b1a1dSMarcel Holtmann continue; 229c08b1a1dSMarcel Holtmann 230470fe1b5SMarcel Holtmann /* Skip the original socket */ 231470fe1b5SMarcel Holtmann if (sk == skip_sk) 232470fe1b5SMarcel Holtmann continue; 233470fe1b5SMarcel Holtmann 234470fe1b5SMarcel Holtmann if (sk->sk_state != BT_BOUND) 235470fe1b5SMarcel Holtmann continue; 236470fe1b5SMarcel Holtmann 2377129069eSJohan Hedberg if (hci_pi(sk)->channel != channel) 238d7f72f61SMarcel Holtmann continue; 239d7f72f61SMarcel Holtmann 240d7f72f61SMarcel Holtmann nskb = skb_clone(skb, GFP_ATOMIC); 241d7f72f61SMarcel Holtmann if (!nskb) 242d7f72f61SMarcel Holtmann continue; 243d7f72f61SMarcel Holtmann 244d7f72f61SMarcel Holtmann if (sock_queue_rcv_skb(sk, nskb)) 245d7f72f61SMarcel Holtmann kfree_skb(nskb); 246d7f72f61SMarcel Holtmann } 247d7f72f61SMarcel Holtmann 248d7f72f61SMarcel Holtmann read_unlock(&hci_sk_list.lock); 249d7f72f61SMarcel Holtmann } 250d7f72f61SMarcel Holtmann 251cd82e61cSMarcel Holtmann /* Send frame to monitor socket */ 252cd82e61cSMarcel Holtmann void hci_send_to_monitor(struct hci_dev *hdev, struct sk_buff *skb) 253cd82e61cSMarcel Holtmann { 254cd82e61cSMarcel Holtmann struct sk_buff *skb_copy = NULL; 2552b531294SMarcel Holtmann struct hci_mon_hdr *hdr; 256cd82e61cSMarcel Holtmann __le16 opcode; 257cd82e61cSMarcel Holtmann 258cd82e61cSMarcel Holtmann if (!atomic_read(&monitor_promisc)) 259cd82e61cSMarcel Holtmann return; 260cd82e61cSMarcel Holtmann 261cd82e61cSMarcel Holtmann BT_DBG("hdev %p len %d", hdev, skb->len); 262cd82e61cSMarcel Holtmann 263cd82e61cSMarcel Holtmann switch (bt_cb(skb)->pkt_type) { 264cd82e61cSMarcel Holtmann case HCI_COMMAND_PKT: 265dcf4adbfSJoe Perches opcode = cpu_to_le16(HCI_MON_COMMAND_PKT); 266cd82e61cSMarcel Holtmann break; 267cd82e61cSMarcel Holtmann case HCI_EVENT_PKT: 268dcf4adbfSJoe Perches opcode = cpu_to_le16(HCI_MON_EVENT_PKT); 269cd82e61cSMarcel Holtmann break; 270cd82e61cSMarcel Holtmann case HCI_ACLDATA_PKT: 271cd82e61cSMarcel Holtmann if (bt_cb(skb)->incoming) 272dcf4adbfSJoe Perches opcode = cpu_to_le16(HCI_MON_ACL_RX_PKT); 273cd82e61cSMarcel Holtmann else 274dcf4adbfSJoe Perches opcode = cpu_to_le16(HCI_MON_ACL_TX_PKT); 275cd82e61cSMarcel Holtmann break; 276cd82e61cSMarcel Holtmann case HCI_SCODATA_PKT: 277cd82e61cSMarcel Holtmann if (bt_cb(skb)->incoming) 278dcf4adbfSJoe Perches opcode = cpu_to_le16(HCI_MON_SCO_RX_PKT); 279cd82e61cSMarcel Holtmann else 280dcf4adbfSJoe Perches opcode = cpu_to_le16(HCI_MON_SCO_TX_PKT); 281cd82e61cSMarcel Holtmann break; 282cd82e61cSMarcel Holtmann default: 283cd82e61cSMarcel Holtmann return; 284cd82e61cSMarcel Holtmann } 285cd82e61cSMarcel Holtmann 2862b531294SMarcel Holtmann /* Create a private copy with headroom */ 2872b531294SMarcel Holtmann skb_copy = __pskb_copy_fclone(skb, HCI_MON_HDR_SIZE, GFP_ATOMIC, true); 2882b531294SMarcel Holtmann if (!skb_copy) 2892b531294SMarcel Holtmann return; 2902b531294SMarcel Holtmann 2912b531294SMarcel Holtmann /* Put header before the data */ 2922b531294SMarcel Holtmann hdr = (void *) skb_push(skb_copy, HCI_MON_HDR_SIZE); 2932b531294SMarcel Holtmann hdr->opcode = opcode; 2942b531294SMarcel Holtmann hdr->index = cpu_to_le16(hdev->id); 2952b531294SMarcel Holtmann hdr->len = cpu_to_le16(skb->len); 2962b531294SMarcel Holtmann 297c08b1a1dSMarcel Holtmann hci_send_to_channel(HCI_CHANNEL_MONITOR, skb_copy, 298c08b1a1dSMarcel Holtmann HCI_SOCK_TRUSTED, NULL); 299cd82e61cSMarcel Holtmann kfree_skb(skb_copy); 300cd82e61cSMarcel Holtmann } 301cd82e61cSMarcel Holtmann 302cd82e61cSMarcel Holtmann static struct sk_buff *create_monitor_event(struct hci_dev *hdev, int event) 303cd82e61cSMarcel Holtmann { 304cd82e61cSMarcel Holtmann struct hci_mon_hdr *hdr; 305cd82e61cSMarcel Holtmann struct hci_mon_new_index *ni; 306cd82e61cSMarcel Holtmann struct sk_buff *skb; 307cd82e61cSMarcel Holtmann __le16 opcode; 308cd82e61cSMarcel Holtmann 309cd82e61cSMarcel Holtmann switch (event) { 310cd82e61cSMarcel Holtmann case HCI_DEV_REG: 311cd82e61cSMarcel Holtmann skb = bt_skb_alloc(HCI_MON_NEW_INDEX_SIZE, GFP_ATOMIC); 312cd82e61cSMarcel Holtmann if (!skb) 313cd82e61cSMarcel Holtmann return NULL; 314cd82e61cSMarcel Holtmann 315cd82e61cSMarcel Holtmann ni = (void *) skb_put(skb, HCI_MON_NEW_INDEX_SIZE); 316cd82e61cSMarcel Holtmann ni->type = hdev->dev_type; 317cd82e61cSMarcel Holtmann ni->bus = hdev->bus; 318cd82e61cSMarcel Holtmann bacpy(&ni->bdaddr, &hdev->bdaddr); 319cd82e61cSMarcel Holtmann memcpy(ni->name, hdev->name, 8); 320cd82e61cSMarcel Holtmann 321dcf4adbfSJoe Perches opcode = cpu_to_le16(HCI_MON_NEW_INDEX); 322cd82e61cSMarcel Holtmann break; 323cd82e61cSMarcel Holtmann 324cd82e61cSMarcel Holtmann case HCI_DEV_UNREG: 325cd82e61cSMarcel Holtmann skb = bt_skb_alloc(0, GFP_ATOMIC); 326cd82e61cSMarcel Holtmann if (!skb) 327cd82e61cSMarcel Holtmann return NULL; 328cd82e61cSMarcel Holtmann 329dcf4adbfSJoe Perches opcode = cpu_to_le16(HCI_MON_DEL_INDEX); 330cd82e61cSMarcel Holtmann break; 331cd82e61cSMarcel Holtmann 332cd82e61cSMarcel Holtmann default: 333cd82e61cSMarcel Holtmann return NULL; 334cd82e61cSMarcel Holtmann } 335cd82e61cSMarcel Holtmann 336cd82e61cSMarcel Holtmann __net_timestamp(skb); 337cd82e61cSMarcel Holtmann 338cd82e61cSMarcel Holtmann hdr = (void *) skb_push(skb, HCI_MON_HDR_SIZE); 339cd82e61cSMarcel Holtmann hdr->opcode = opcode; 340cd82e61cSMarcel Holtmann hdr->index = cpu_to_le16(hdev->id); 341cd82e61cSMarcel Holtmann hdr->len = cpu_to_le16(skb->len - HCI_MON_HDR_SIZE); 342cd82e61cSMarcel Holtmann 343cd82e61cSMarcel Holtmann return skb; 344cd82e61cSMarcel Holtmann } 345cd82e61cSMarcel Holtmann 346cd82e61cSMarcel Holtmann static void send_monitor_replay(struct sock *sk) 347cd82e61cSMarcel Holtmann { 348cd82e61cSMarcel Holtmann struct hci_dev *hdev; 349cd82e61cSMarcel Holtmann 350cd82e61cSMarcel Holtmann read_lock(&hci_dev_list_lock); 351cd82e61cSMarcel Holtmann 352cd82e61cSMarcel Holtmann list_for_each_entry(hdev, &hci_dev_list, list) { 353cd82e61cSMarcel Holtmann struct sk_buff *skb; 354cd82e61cSMarcel Holtmann 355cd82e61cSMarcel Holtmann skb = create_monitor_event(hdev, HCI_DEV_REG); 356cd82e61cSMarcel Holtmann if (!skb) 357cd82e61cSMarcel Holtmann continue; 358cd82e61cSMarcel Holtmann 359cd82e61cSMarcel Holtmann if (sock_queue_rcv_skb(sk, skb)) 360cd82e61cSMarcel Holtmann kfree_skb(skb); 361cd82e61cSMarcel Holtmann } 362cd82e61cSMarcel Holtmann 363cd82e61cSMarcel Holtmann read_unlock(&hci_dev_list_lock); 364cd82e61cSMarcel Holtmann } 365cd82e61cSMarcel Holtmann 366040030efSMarcel Holtmann /* Generate internal stack event */ 367040030efSMarcel Holtmann static void hci_si_event(struct hci_dev *hdev, int type, int dlen, void *data) 368040030efSMarcel Holtmann { 369040030efSMarcel Holtmann struct hci_event_hdr *hdr; 370040030efSMarcel Holtmann struct hci_ev_stack_internal *ev; 371040030efSMarcel Holtmann struct sk_buff *skb; 372040030efSMarcel Holtmann 373040030efSMarcel Holtmann skb = bt_skb_alloc(HCI_EVENT_HDR_SIZE + sizeof(*ev) + dlen, GFP_ATOMIC); 374040030efSMarcel Holtmann if (!skb) 375040030efSMarcel Holtmann return; 376040030efSMarcel Holtmann 377040030efSMarcel Holtmann hdr = (void *) skb_put(skb, HCI_EVENT_HDR_SIZE); 378040030efSMarcel Holtmann hdr->evt = HCI_EV_STACK_INTERNAL; 379040030efSMarcel Holtmann hdr->plen = sizeof(*ev) + dlen; 380040030efSMarcel Holtmann 381040030efSMarcel Holtmann ev = (void *) skb_put(skb, sizeof(*ev) + dlen); 382040030efSMarcel Holtmann ev->type = type; 383040030efSMarcel Holtmann memcpy(ev->data, data, dlen); 384040030efSMarcel Holtmann 385040030efSMarcel Holtmann bt_cb(skb)->incoming = 1; 386040030efSMarcel Holtmann __net_timestamp(skb); 387040030efSMarcel Holtmann 388040030efSMarcel Holtmann bt_cb(skb)->pkt_type = HCI_EVENT_PKT; 389040030efSMarcel Holtmann hci_send_to_sock(hdev, skb); 390040030efSMarcel Holtmann kfree_skb(skb); 391040030efSMarcel Holtmann } 392040030efSMarcel Holtmann 393040030efSMarcel Holtmann void hci_sock_dev_event(struct hci_dev *hdev, int event) 394040030efSMarcel Holtmann { 395040030efSMarcel Holtmann struct hci_ev_si_device ev; 396040030efSMarcel Holtmann 397040030efSMarcel Holtmann BT_DBG("hdev %s event %d", hdev->name, event); 398040030efSMarcel Holtmann 399cd82e61cSMarcel Holtmann /* Send event to monitor */ 400cd82e61cSMarcel Holtmann if (atomic_read(&monitor_promisc)) { 401cd82e61cSMarcel Holtmann struct sk_buff *skb; 402cd82e61cSMarcel Holtmann 403cd82e61cSMarcel Holtmann skb = create_monitor_event(hdev, event); 404cd82e61cSMarcel Holtmann if (skb) { 405c08b1a1dSMarcel Holtmann hci_send_to_channel(HCI_CHANNEL_MONITOR, skb, 406c08b1a1dSMarcel Holtmann HCI_SOCK_TRUSTED, NULL); 407cd82e61cSMarcel Holtmann kfree_skb(skb); 408cd82e61cSMarcel Holtmann } 409cd82e61cSMarcel Holtmann } 410cd82e61cSMarcel Holtmann 411040030efSMarcel Holtmann /* Send event to sockets */ 412040030efSMarcel Holtmann ev.event = event; 413040030efSMarcel Holtmann ev.dev_id = hdev->id; 414040030efSMarcel Holtmann hci_si_event(NULL, HCI_EV_SI_DEVICE, sizeof(ev), &ev); 415040030efSMarcel Holtmann 416040030efSMarcel Holtmann if (event == HCI_DEV_UNREG) { 417040030efSMarcel Holtmann struct sock *sk; 418040030efSMarcel Holtmann 419040030efSMarcel Holtmann /* Detach sockets from device */ 420040030efSMarcel Holtmann read_lock(&hci_sk_list.lock); 421b67bfe0dSSasha Levin sk_for_each(sk, &hci_sk_list.head) { 422040030efSMarcel Holtmann bh_lock_sock_nested(sk); 423040030efSMarcel Holtmann if (hci_pi(sk)->hdev == hdev) { 424040030efSMarcel Holtmann hci_pi(sk)->hdev = NULL; 425040030efSMarcel Holtmann sk->sk_err = EPIPE; 426040030efSMarcel Holtmann sk->sk_state = BT_OPEN; 427040030efSMarcel Holtmann sk->sk_state_change(sk); 428040030efSMarcel Holtmann 429040030efSMarcel Holtmann hci_dev_put(hdev); 430040030efSMarcel Holtmann } 431040030efSMarcel Holtmann bh_unlock_sock(sk); 432040030efSMarcel Holtmann } 433040030efSMarcel Holtmann read_unlock(&hci_sk_list.lock); 434040030efSMarcel Holtmann } 435040030efSMarcel Holtmann } 436040030efSMarcel Holtmann 437801c1e8dSJohan Hedberg static struct hci_mgmt_chan *__hci_mgmt_chan_find(unsigned short channel) 438801c1e8dSJohan Hedberg { 439801c1e8dSJohan Hedberg struct hci_mgmt_chan *c; 440801c1e8dSJohan Hedberg 441801c1e8dSJohan Hedberg list_for_each_entry(c, &mgmt_chan_list, list) { 442801c1e8dSJohan Hedberg if (c->channel == channel) 443801c1e8dSJohan Hedberg return c; 444801c1e8dSJohan Hedberg } 445801c1e8dSJohan Hedberg 446801c1e8dSJohan Hedberg return NULL; 447801c1e8dSJohan Hedberg } 448801c1e8dSJohan Hedberg 449801c1e8dSJohan Hedberg static struct hci_mgmt_chan *hci_mgmt_chan_find(unsigned short channel) 450801c1e8dSJohan Hedberg { 451801c1e8dSJohan Hedberg struct hci_mgmt_chan *c; 452801c1e8dSJohan Hedberg 453801c1e8dSJohan Hedberg mutex_lock(&mgmt_chan_list_lock); 454801c1e8dSJohan Hedberg c = __hci_mgmt_chan_find(channel); 455801c1e8dSJohan Hedberg mutex_unlock(&mgmt_chan_list_lock); 456801c1e8dSJohan Hedberg 457801c1e8dSJohan Hedberg return c; 458801c1e8dSJohan Hedberg } 459801c1e8dSJohan Hedberg 460801c1e8dSJohan Hedberg int hci_mgmt_chan_register(struct hci_mgmt_chan *c) 461801c1e8dSJohan Hedberg { 462801c1e8dSJohan Hedberg if (c->channel < HCI_CHANNEL_CONTROL) 463801c1e8dSJohan Hedberg return -EINVAL; 464801c1e8dSJohan Hedberg 465801c1e8dSJohan Hedberg mutex_lock(&mgmt_chan_list_lock); 466801c1e8dSJohan Hedberg if (__hci_mgmt_chan_find(c->channel)) { 467801c1e8dSJohan Hedberg mutex_unlock(&mgmt_chan_list_lock); 468801c1e8dSJohan Hedberg return -EALREADY; 469801c1e8dSJohan Hedberg } 470801c1e8dSJohan Hedberg 471801c1e8dSJohan Hedberg list_add_tail(&c->list, &mgmt_chan_list); 472801c1e8dSJohan Hedberg 473801c1e8dSJohan Hedberg mutex_unlock(&mgmt_chan_list_lock); 474801c1e8dSJohan Hedberg 475801c1e8dSJohan Hedberg return 0; 476801c1e8dSJohan Hedberg } 477801c1e8dSJohan Hedberg EXPORT_SYMBOL(hci_mgmt_chan_register); 478801c1e8dSJohan Hedberg 479801c1e8dSJohan Hedberg void hci_mgmt_chan_unregister(struct hci_mgmt_chan *c) 480801c1e8dSJohan Hedberg { 481801c1e8dSJohan Hedberg mutex_lock(&mgmt_chan_list_lock); 482801c1e8dSJohan Hedberg list_del(&c->list); 483801c1e8dSJohan Hedberg mutex_unlock(&mgmt_chan_list_lock); 484801c1e8dSJohan Hedberg } 485801c1e8dSJohan Hedberg EXPORT_SYMBOL(hci_mgmt_chan_unregister); 486801c1e8dSJohan Hedberg 4871da177e4SLinus Torvalds static int hci_sock_release(struct socket *sock) 4881da177e4SLinus Torvalds { 4891da177e4SLinus Torvalds struct sock *sk = sock->sk; 4907b005bd3SMarcel Holtmann struct hci_dev *hdev; 4911da177e4SLinus Torvalds 4921da177e4SLinus Torvalds BT_DBG("sock %p sk %p", sock, sk); 4931da177e4SLinus Torvalds 4941da177e4SLinus Torvalds if (!sk) 4951da177e4SLinus Torvalds return 0; 4961da177e4SLinus Torvalds 4977b005bd3SMarcel Holtmann hdev = hci_pi(sk)->hdev; 4987b005bd3SMarcel Holtmann 499cd82e61cSMarcel Holtmann if (hci_pi(sk)->channel == HCI_CHANNEL_MONITOR) 500cd82e61cSMarcel Holtmann atomic_dec(&monitor_promisc); 501cd82e61cSMarcel Holtmann 5021da177e4SLinus Torvalds bt_sock_unlink(&hci_sk_list, sk); 5031da177e4SLinus Torvalds 5041da177e4SLinus Torvalds if (hdev) { 50523500189SMarcel Holtmann if (hci_pi(sk)->channel == HCI_CHANNEL_USER) { 506*6b3cc1dbSSimon Fels /* When releasing an user channel exclusive access, 507*6b3cc1dbSSimon Fels * call hci_dev_do_close directly instead of calling 508*6b3cc1dbSSimon Fels * hci_dev_close to ensure the exclusive access will 509*6b3cc1dbSSimon Fels * be released and the controller brought back down. 510*6b3cc1dbSSimon Fels * 511*6b3cc1dbSSimon Fels * The checking of HCI_AUTO_OFF is not needed in this 512*6b3cc1dbSSimon Fels * case since it will have been cleared already when 513*6b3cc1dbSSimon Fels * opening the user channel. 514*6b3cc1dbSSimon Fels */ 515*6b3cc1dbSSimon Fels hci_dev_do_close(hdev); 5169380f9eaSLoic Poulain hci_dev_clear_flag(hdev, HCI_USER_CHANNEL); 5179380f9eaSLoic Poulain mgmt_index_added(hdev); 51823500189SMarcel Holtmann } 51923500189SMarcel Holtmann 5201da177e4SLinus Torvalds atomic_dec(&hdev->promisc); 5211da177e4SLinus Torvalds hci_dev_put(hdev); 5221da177e4SLinus Torvalds } 5231da177e4SLinus Torvalds 5241da177e4SLinus Torvalds sock_orphan(sk); 5251da177e4SLinus Torvalds 5261da177e4SLinus Torvalds skb_queue_purge(&sk->sk_receive_queue); 5271da177e4SLinus Torvalds skb_queue_purge(&sk->sk_write_queue); 5281da177e4SLinus Torvalds 5291da177e4SLinus Torvalds sock_put(sk); 5301da177e4SLinus Torvalds return 0; 5311da177e4SLinus Torvalds } 5321da177e4SLinus Torvalds 533b2a66aadSAntti Julku static int hci_sock_blacklist_add(struct hci_dev *hdev, void __user *arg) 534f0358568SJohan Hedberg { 535f0358568SJohan Hedberg bdaddr_t bdaddr; 5365e762444SAntti Julku int err; 537f0358568SJohan Hedberg 538f0358568SJohan Hedberg if (copy_from_user(&bdaddr, arg, sizeof(bdaddr))) 539f0358568SJohan Hedberg return -EFAULT; 540f0358568SJohan Hedberg 54109fd0de5SGustavo F. Padovan hci_dev_lock(hdev); 5425e762444SAntti Julku 543dcc36c16SJohan Hedberg err = hci_bdaddr_list_add(&hdev->blacklist, &bdaddr, BDADDR_BREDR); 5445e762444SAntti Julku 54509fd0de5SGustavo F. Padovan hci_dev_unlock(hdev); 5465e762444SAntti Julku 5475e762444SAntti Julku return err; 548f0358568SJohan Hedberg } 549f0358568SJohan Hedberg 550b2a66aadSAntti Julku static int hci_sock_blacklist_del(struct hci_dev *hdev, void __user *arg) 551f0358568SJohan Hedberg { 552f0358568SJohan Hedberg bdaddr_t bdaddr; 5535e762444SAntti Julku int err; 554f0358568SJohan Hedberg 555f0358568SJohan Hedberg if (copy_from_user(&bdaddr, arg, sizeof(bdaddr))) 556f0358568SJohan Hedberg return -EFAULT; 557f0358568SJohan Hedberg 55809fd0de5SGustavo F. Padovan hci_dev_lock(hdev); 5595e762444SAntti Julku 560dcc36c16SJohan Hedberg err = hci_bdaddr_list_del(&hdev->blacklist, &bdaddr, BDADDR_BREDR); 5615e762444SAntti Julku 56209fd0de5SGustavo F. Padovan hci_dev_unlock(hdev); 5635e762444SAntti Julku 5645e762444SAntti Julku return err; 565f0358568SJohan Hedberg } 566f0358568SJohan Hedberg 5671da177e4SLinus Torvalds /* Ioctls that require bound socket */ 5686039aa73SGustavo Padovan static int hci_sock_bound_ioctl(struct sock *sk, unsigned int cmd, 5696039aa73SGustavo Padovan unsigned long arg) 5701da177e4SLinus Torvalds { 5711da177e4SLinus Torvalds struct hci_dev *hdev = hci_pi(sk)->hdev; 5721da177e4SLinus Torvalds 5731da177e4SLinus Torvalds if (!hdev) 5741da177e4SLinus Torvalds return -EBADFD; 5751da177e4SLinus Torvalds 576d7a5a11dSMarcel Holtmann if (hci_dev_test_flag(hdev, HCI_USER_CHANNEL)) 5770736cfa8SMarcel Holtmann return -EBUSY; 5780736cfa8SMarcel Holtmann 579d7a5a11dSMarcel Holtmann if (hci_dev_test_flag(hdev, HCI_UNCONFIGURED)) 580fee746b0SMarcel Holtmann return -EOPNOTSUPP; 581fee746b0SMarcel Holtmann 5825b69bef5SMarcel Holtmann if (hdev->dev_type != HCI_BREDR) 5835b69bef5SMarcel Holtmann return -EOPNOTSUPP; 5845b69bef5SMarcel Holtmann 5851da177e4SLinus Torvalds switch (cmd) { 5861da177e4SLinus Torvalds case HCISETRAW: 5871da177e4SLinus Torvalds if (!capable(CAP_NET_ADMIN)) 588bf5b30b8SZhao Hongjiang return -EPERM; 589db596681SMarcel Holtmann return -EOPNOTSUPP; 5901da177e4SLinus Torvalds 5911da177e4SLinus Torvalds case HCIGETCONNINFO: 5921da177e4SLinus Torvalds return hci_get_conn_info(hdev, (void __user *) arg); 5931da177e4SLinus Torvalds 59440be492fSMarcel Holtmann case HCIGETAUTHINFO: 59540be492fSMarcel Holtmann return hci_get_auth_info(hdev, (void __user *) arg); 59640be492fSMarcel Holtmann 597f0358568SJohan Hedberg case HCIBLOCKADDR: 598f0358568SJohan Hedberg if (!capable(CAP_NET_ADMIN)) 599bf5b30b8SZhao Hongjiang return -EPERM; 600b2a66aadSAntti Julku return hci_sock_blacklist_add(hdev, (void __user *) arg); 601f0358568SJohan Hedberg 602f0358568SJohan Hedberg case HCIUNBLOCKADDR: 603f0358568SJohan Hedberg if (!capable(CAP_NET_ADMIN)) 604bf5b30b8SZhao Hongjiang return -EPERM; 605b2a66aadSAntti Julku return hci_sock_blacklist_del(hdev, (void __user *) arg); 6060736cfa8SMarcel Holtmann } 607f0358568SJohan Hedberg 608324d36edSMarcel Holtmann return -ENOIOCTLCMD; 6091da177e4SLinus Torvalds } 6101da177e4SLinus Torvalds 6118fc9ced3SGustavo Padovan static int hci_sock_ioctl(struct socket *sock, unsigned int cmd, 6128fc9ced3SGustavo Padovan unsigned long arg) 6131da177e4SLinus Torvalds { 6141da177e4SLinus Torvalds void __user *argp = (void __user *) arg; 6150736cfa8SMarcel Holtmann struct sock *sk = sock->sk; 6161da177e4SLinus Torvalds int err; 6171da177e4SLinus Torvalds 6181da177e4SLinus Torvalds BT_DBG("cmd %x arg %lx", cmd, arg); 6191da177e4SLinus Torvalds 620c1c4f956SMarcel Holtmann lock_sock(sk); 621c1c4f956SMarcel Holtmann 622c1c4f956SMarcel Holtmann if (hci_pi(sk)->channel != HCI_CHANNEL_RAW) { 623c1c4f956SMarcel Holtmann err = -EBADFD; 624c1c4f956SMarcel Holtmann goto done; 625c1c4f956SMarcel Holtmann } 626c1c4f956SMarcel Holtmann 627c1c4f956SMarcel Holtmann release_sock(sk); 628c1c4f956SMarcel Holtmann 6291da177e4SLinus Torvalds switch (cmd) { 6301da177e4SLinus Torvalds case HCIGETDEVLIST: 6311da177e4SLinus Torvalds return hci_get_dev_list(argp); 6321da177e4SLinus Torvalds 6331da177e4SLinus Torvalds case HCIGETDEVINFO: 6341da177e4SLinus Torvalds return hci_get_dev_info(argp); 6351da177e4SLinus Torvalds 6361da177e4SLinus Torvalds case HCIGETCONNLIST: 6371da177e4SLinus Torvalds return hci_get_conn_list(argp); 6381da177e4SLinus Torvalds 6391da177e4SLinus Torvalds case HCIDEVUP: 6401da177e4SLinus Torvalds if (!capable(CAP_NET_ADMIN)) 641bf5b30b8SZhao Hongjiang return -EPERM; 6421da177e4SLinus Torvalds return hci_dev_open(arg); 6431da177e4SLinus Torvalds 6441da177e4SLinus Torvalds case HCIDEVDOWN: 6451da177e4SLinus Torvalds if (!capable(CAP_NET_ADMIN)) 646bf5b30b8SZhao Hongjiang return -EPERM; 6471da177e4SLinus Torvalds return hci_dev_close(arg); 6481da177e4SLinus Torvalds 6491da177e4SLinus Torvalds case HCIDEVRESET: 6501da177e4SLinus Torvalds if (!capable(CAP_NET_ADMIN)) 651bf5b30b8SZhao Hongjiang return -EPERM; 6521da177e4SLinus Torvalds return hci_dev_reset(arg); 6531da177e4SLinus Torvalds 6541da177e4SLinus Torvalds case HCIDEVRESTAT: 6551da177e4SLinus Torvalds if (!capable(CAP_NET_ADMIN)) 656bf5b30b8SZhao Hongjiang return -EPERM; 6571da177e4SLinus Torvalds return hci_dev_reset_stat(arg); 6581da177e4SLinus Torvalds 6591da177e4SLinus Torvalds case HCISETSCAN: 6601da177e4SLinus Torvalds case HCISETAUTH: 6611da177e4SLinus Torvalds case HCISETENCRYPT: 6621da177e4SLinus Torvalds case HCISETPTYPE: 6631da177e4SLinus Torvalds case HCISETLINKPOL: 6641da177e4SLinus Torvalds case HCISETLINKMODE: 6651da177e4SLinus Torvalds case HCISETACLMTU: 6661da177e4SLinus Torvalds case HCISETSCOMTU: 6671da177e4SLinus Torvalds if (!capable(CAP_NET_ADMIN)) 668bf5b30b8SZhao Hongjiang return -EPERM; 6691da177e4SLinus Torvalds return hci_dev_cmd(cmd, argp); 6701da177e4SLinus Torvalds 6711da177e4SLinus Torvalds case HCIINQUIRY: 6721da177e4SLinus Torvalds return hci_inquiry(argp); 673c1c4f956SMarcel Holtmann } 6741da177e4SLinus Torvalds 6751da177e4SLinus Torvalds lock_sock(sk); 676c1c4f956SMarcel Holtmann 6771da177e4SLinus Torvalds err = hci_sock_bound_ioctl(sk, cmd, arg); 678c1c4f956SMarcel Holtmann 679c1c4f956SMarcel Holtmann done: 6801da177e4SLinus Torvalds release_sock(sk); 6811da177e4SLinus Torvalds return err; 6821da177e4SLinus Torvalds } 6831da177e4SLinus Torvalds 6848fc9ced3SGustavo Padovan static int hci_sock_bind(struct socket *sock, struct sockaddr *addr, 6858fc9ced3SGustavo Padovan int addr_len) 6861da177e4SLinus Torvalds { 6870381101fSJohan Hedberg struct sockaddr_hci haddr; 6881da177e4SLinus Torvalds struct sock *sk = sock->sk; 6891da177e4SLinus Torvalds struct hci_dev *hdev = NULL; 6900381101fSJohan Hedberg int len, err = 0; 6911da177e4SLinus Torvalds 6921da177e4SLinus Torvalds BT_DBG("sock %p sk %p", sock, sk); 6931da177e4SLinus Torvalds 6940381101fSJohan Hedberg if (!addr) 6950381101fSJohan Hedberg return -EINVAL; 6960381101fSJohan Hedberg 6970381101fSJohan Hedberg memset(&haddr, 0, sizeof(haddr)); 6980381101fSJohan Hedberg len = min_t(unsigned int, sizeof(haddr), addr_len); 6990381101fSJohan Hedberg memcpy(&haddr, addr, len); 7000381101fSJohan Hedberg 7010381101fSJohan Hedberg if (haddr.hci_family != AF_BLUETOOTH) 7020381101fSJohan Hedberg return -EINVAL; 7030381101fSJohan Hedberg 7041da177e4SLinus Torvalds lock_sock(sk); 7051da177e4SLinus Torvalds 7067cc2ade2SMarcel Holtmann if (sk->sk_state == BT_BOUND) { 7077cc2ade2SMarcel Holtmann err = -EALREADY; 7087cc2ade2SMarcel Holtmann goto done; 7097cc2ade2SMarcel Holtmann } 7107cc2ade2SMarcel Holtmann 7117cc2ade2SMarcel Holtmann switch (haddr.hci_channel) { 7127cc2ade2SMarcel Holtmann case HCI_CHANNEL_RAW: 7137cc2ade2SMarcel Holtmann if (hci_pi(sk)->hdev) { 7141da177e4SLinus Torvalds err = -EALREADY; 7151da177e4SLinus Torvalds goto done; 7161da177e4SLinus Torvalds } 7171da177e4SLinus Torvalds 7180381101fSJohan Hedberg if (haddr.hci_dev != HCI_DEV_NONE) { 7190381101fSJohan Hedberg hdev = hci_dev_get(haddr.hci_dev); 72070f23020SAndrei Emeltchenko if (!hdev) { 7211da177e4SLinus Torvalds err = -ENODEV; 7221da177e4SLinus Torvalds goto done; 7231da177e4SLinus Torvalds } 7241da177e4SLinus Torvalds 7251da177e4SLinus Torvalds atomic_inc(&hdev->promisc); 7261da177e4SLinus Torvalds } 7271da177e4SLinus Torvalds 7281da177e4SLinus Torvalds hci_pi(sk)->hdev = hdev; 7297cc2ade2SMarcel Holtmann break; 7307cc2ade2SMarcel Holtmann 73123500189SMarcel Holtmann case HCI_CHANNEL_USER: 73223500189SMarcel Holtmann if (hci_pi(sk)->hdev) { 73323500189SMarcel Holtmann err = -EALREADY; 73423500189SMarcel Holtmann goto done; 73523500189SMarcel Holtmann } 73623500189SMarcel Holtmann 73723500189SMarcel Holtmann if (haddr.hci_dev == HCI_DEV_NONE) { 73823500189SMarcel Holtmann err = -EINVAL; 73923500189SMarcel Holtmann goto done; 74023500189SMarcel Holtmann } 74123500189SMarcel Holtmann 74210a8b86fSMarcel Holtmann if (!capable(CAP_NET_ADMIN)) { 74323500189SMarcel Holtmann err = -EPERM; 74423500189SMarcel Holtmann goto done; 74523500189SMarcel Holtmann } 74623500189SMarcel Holtmann 74723500189SMarcel Holtmann hdev = hci_dev_get(haddr.hci_dev); 74823500189SMarcel Holtmann if (!hdev) { 74923500189SMarcel Holtmann err = -ENODEV; 75023500189SMarcel Holtmann goto done; 75123500189SMarcel Holtmann } 75223500189SMarcel Holtmann 753781f899fSMarcel Holtmann if (test_bit(HCI_INIT, &hdev->flags) || 754d7a5a11dSMarcel Holtmann hci_dev_test_flag(hdev, HCI_SETUP) || 755781f899fSMarcel Holtmann hci_dev_test_flag(hdev, HCI_CONFIG) || 756781f899fSMarcel Holtmann (!hci_dev_test_flag(hdev, HCI_AUTO_OFF) && 757781f899fSMarcel Holtmann test_bit(HCI_UP, &hdev->flags))) { 75823500189SMarcel Holtmann err = -EBUSY; 75923500189SMarcel Holtmann hci_dev_put(hdev); 76023500189SMarcel Holtmann goto done; 76123500189SMarcel Holtmann } 76223500189SMarcel Holtmann 763238be788SMarcel Holtmann if (hci_dev_test_and_set_flag(hdev, HCI_USER_CHANNEL)) { 76423500189SMarcel Holtmann err = -EUSERS; 76523500189SMarcel Holtmann hci_dev_put(hdev); 76623500189SMarcel Holtmann goto done; 76723500189SMarcel Holtmann } 76823500189SMarcel Holtmann 76923500189SMarcel Holtmann mgmt_index_removed(hdev); 77023500189SMarcel Holtmann 77123500189SMarcel Holtmann err = hci_dev_open(hdev->id); 77223500189SMarcel Holtmann if (err) { 773781f899fSMarcel Holtmann if (err == -EALREADY) { 774781f899fSMarcel Holtmann /* In case the transport is already up and 775781f899fSMarcel Holtmann * running, clear the error here. 776781f899fSMarcel Holtmann * 777781f899fSMarcel Holtmann * This can happen when opening an user 778781f899fSMarcel Holtmann * channel and HCI_AUTO_OFF grace period 779781f899fSMarcel Holtmann * is still active. 780781f899fSMarcel Holtmann */ 781781f899fSMarcel Holtmann err = 0; 782781f899fSMarcel Holtmann } else { 783a358dc11SMarcel Holtmann hci_dev_clear_flag(hdev, HCI_USER_CHANNEL); 784c6521401SMarcel Holtmann mgmt_index_added(hdev); 78523500189SMarcel Holtmann hci_dev_put(hdev); 78623500189SMarcel Holtmann goto done; 78723500189SMarcel Holtmann } 788781f899fSMarcel Holtmann } 78923500189SMarcel Holtmann 79023500189SMarcel Holtmann atomic_inc(&hdev->promisc); 79123500189SMarcel Holtmann 79223500189SMarcel Holtmann hci_pi(sk)->hdev = hdev; 79323500189SMarcel Holtmann break; 79423500189SMarcel Holtmann 795cd82e61cSMarcel Holtmann case HCI_CHANNEL_MONITOR: 796cd82e61cSMarcel Holtmann if (haddr.hci_dev != HCI_DEV_NONE) { 797cd82e61cSMarcel Holtmann err = -EINVAL; 798cd82e61cSMarcel Holtmann goto done; 799cd82e61cSMarcel Holtmann } 800cd82e61cSMarcel Holtmann 801cd82e61cSMarcel Holtmann if (!capable(CAP_NET_RAW)) { 802cd82e61cSMarcel Holtmann err = -EPERM; 803cd82e61cSMarcel Holtmann goto done; 804cd82e61cSMarcel Holtmann } 805cd82e61cSMarcel Holtmann 80650ebc055SMarcel Holtmann /* The monitor interface is restricted to CAP_NET_RAW 80750ebc055SMarcel Holtmann * capabilities and with that implicitly trusted. 80850ebc055SMarcel Holtmann */ 80950ebc055SMarcel Holtmann hci_sock_set_flag(sk, HCI_SOCK_TRUSTED); 81050ebc055SMarcel Holtmann 811cd82e61cSMarcel Holtmann send_monitor_replay(sk); 812cd82e61cSMarcel Holtmann 813cd82e61cSMarcel Holtmann atomic_inc(&monitor_promisc); 814cd82e61cSMarcel Holtmann break; 815cd82e61cSMarcel Holtmann 8167cc2ade2SMarcel Holtmann default: 817801c1e8dSJohan Hedberg if (!hci_mgmt_chan_find(haddr.hci_channel)) { 8187cc2ade2SMarcel Holtmann err = -EINVAL; 8197cc2ade2SMarcel Holtmann goto done; 8207cc2ade2SMarcel Holtmann } 8217cc2ade2SMarcel Holtmann 822801c1e8dSJohan Hedberg if (haddr.hci_dev != HCI_DEV_NONE) { 823801c1e8dSJohan Hedberg err = -EINVAL; 824801c1e8dSJohan Hedberg goto done; 825801c1e8dSJohan Hedberg } 826801c1e8dSJohan Hedberg 8271195fbb8SMarcel Holtmann /* Users with CAP_NET_ADMIN capabilities are allowed 8281195fbb8SMarcel Holtmann * access to all management commands and events. For 8291195fbb8SMarcel Holtmann * untrusted users the interface is restricted and 8301195fbb8SMarcel Holtmann * also only untrusted events are sent. 83150ebc055SMarcel Holtmann */ 8321195fbb8SMarcel Holtmann if (capable(CAP_NET_ADMIN)) 83350ebc055SMarcel Holtmann hci_sock_set_flag(sk, HCI_SOCK_TRUSTED); 83450ebc055SMarcel Holtmann 835f9207338SMarcel Holtmann /* At the moment the index and unconfigured index events 836f9207338SMarcel Holtmann * are enabled unconditionally. Setting them on each 837f9207338SMarcel Holtmann * socket when binding keeps this functionality. They 838f9207338SMarcel Holtmann * however might be cleared later and then sending of these 839f9207338SMarcel Holtmann * events will be disabled, but that is then intentional. 840f6b7712eSMarcel Holtmann * 841f6b7712eSMarcel Holtmann * This also enables generic events that are safe to be 842f6b7712eSMarcel Holtmann * received by untrusted users. Example for such events 843f6b7712eSMarcel Holtmann * are changes to settings, class of device, name etc. 844f9207338SMarcel Holtmann */ 845f9207338SMarcel Holtmann if (haddr.hci_channel == HCI_CHANNEL_CONTROL) { 846f9207338SMarcel Holtmann hci_sock_set_flag(sk, HCI_MGMT_INDEX_EVENTS); 847f9207338SMarcel Holtmann hci_sock_set_flag(sk, HCI_MGMT_UNCONF_INDEX_EVENTS); 848f6b7712eSMarcel Holtmann hci_sock_set_flag(sk, HCI_MGMT_GENERIC_EVENTS); 849f9207338SMarcel Holtmann } 850801c1e8dSJohan Hedberg break; 851801c1e8dSJohan Hedberg } 852801c1e8dSJohan Hedberg 8537cc2ade2SMarcel Holtmann 8547cc2ade2SMarcel Holtmann hci_pi(sk)->channel = haddr.hci_channel; 8551da177e4SLinus Torvalds sk->sk_state = BT_BOUND; 8561da177e4SLinus Torvalds 8571da177e4SLinus Torvalds done: 8581da177e4SLinus Torvalds release_sock(sk); 8591da177e4SLinus Torvalds return err; 8601da177e4SLinus Torvalds } 8611da177e4SLinus Torvalds 8628fc9ced3SGustavo Padovan static int hci_sock_getname(struct socket *sock, struct sockaddr *addr, 8638fc9ced3SGustavo Padovan int *addr_len, int peer) 8641da177e4SLinus Torvalds { 8651da177e4SLinus Torvalds struct sockaddr_hci *haddr = (struct sockaddr_hci *) addr; 8661da177e4SLinus Torvalds struct sock *sk = sock->sk; 8679d4b68b2SMarcel Holtmann struct hci_dev *hdev; 8689d4b68b2SMarcel Holtmann int err = 0; 8691da177e4SLinus Torvalds 8701da177e4SLinus Torvalds BT_DBG("sock %p sk %p", sock, sk); 8711da177e4SLinus Torvalds 87206f43cbcSMarcel Holtmann if (peer) 87306f43cbcSMarcel Holtmann return -EOPNOTSUPP; 87406f43cbcSMarcel Holtmann 8751da177e4SLinus Torvalds lock_sock(sk); 8761da177e4SLinus Torvalds 8779d4b68b2SMarcel Holtmann hdev = hci_pi(sk)->hdev; 8789d4b68b2SMarcel Holtmann if (!hdev) { 8799d4b68b2SMarcel Holtmann err = -EBADFD; 8809d4b68b2SMarcel Holtmann goto done; 8819d4b68b2SMarcel Holtmann } 8829d4b68b2SMarcel Holtmann 8831da177e4SLinus Torvalds *addr_len = sizeof(*haddr); 8841da177e4SLinus Torvalds haddr->hci_family = AF_BLUETOOTH; 8857b005bd3SMarcel Holtmann haddr->hci_dev = hdev->id; 8869d4b68b2SMarcel Holtmann haddr->hci_channel= hci_pi(sk)->channel; 8871da177e4SLinus Torvalds 8889d4b68b2SMarcel Holtmann done: 8891da177e4SLinus Torvalds release_sock(sk); 8909d4b68b2SMarcel Holtmann return err; 8911da177e4SLinus Torvalds } 8921da177e4SLinus Torvalds 8936039aa73SGustavo Padovan static void hci_sock_cmsg(struct sock *sk, struct msghdr *msg, 8946039aa73SGustavo Padovan struct sk_buff *skb) 8951da177e4SLinus Torvalds { 8961da177e4SLinus Torvalds __u32 mask = hci_pi(sk)->cmsg_mask; 8971da177e4SLinus Torvalds 8980d48d939SMarcel Holtmann if (mask & HCI_CMSG_DIR) { 8990d48d939SMarcel Holtmann int incoming = bt_cb(skb)->incoming; 9008fc9ced3SGustavo Padovan put_cmsg(msg, SOL_HCI, HCI_CMSG_DIR, sizeof(incoming), 9018fc9ced3SGustavo Padovan &incoming); 9020d48d939SMarcel Holtmann } 9031da177e4SLinus Torvalds 904a61bbcf2SPatrick McHardy if (mask & HCI_CMSG_TSTAMP) { 905f6e623a6SJohann Felix Soden #ifdef CONFIG_COMPAT 906f6e623a6SJohann Felix Soden struct compat_timeval ctv; 907f6e623a6SJohann Felix Soden #endif 908a61bbcf2SPatrick McHardy struct timeval tv; 909767c5eb5SMarcel Holtmann void *data; 910767c5eb5SMarcel Holtmann int len; 911a61bbcf2SPatrick McHardy 912a61bbcf2SPatrick McHardy skb_get_timestamp(skb, &tv); 913767c5eb5SMarcel Holtmann 9141da97f83SDavid S. Miller data = &tv; 9151da97f83SDavid S. Miller len = sizeof(tv); 9161da97f83SDavid S. Miller #ifdef CONFIG_COMPAT 917da88cea1SH. J. Lu if (!COMPAT_USE_64BIT_TIME && 918da88cea1SH. J. Lu (msg->msg_flags & MSG_CMSG_COMPAT)) { 919767c5eb5SMarcel Holtmann ctv.tv_sec = tv.tv_sec; 920767c5eb5SMarcel Holtmann ctv.tv_usec = tv.tv_usec; 921767c5eb5SMarcel Holtmann data = &ctv; 922767c5eb5SMarcel Holtmann len = sizeof(ctv); 923767c5eb5SMarcel Holtmann } 9241da97f83SDavid S. Miller #endif 925767c5eb5SMarcel Holtmann 926767c5eb5SMarcel Holtmann put_cmsg(msg, SOL_HCI, HCI_CMSG_TSTAMP, len, data); 927a61bbcf2SPatrick McHardy } 9281da177e4SLinus Torvalds } 9291da177e4SLinus Torvalds 9301b784140SYing Xue static int hci_sock_recvmsg(struct socket *sock, struct msghdr *msg, size_t len, 9311b784140SYing Xue int flags) 9321da177e4SLinus Torvalds { 9331da177e4SLinus Torvalds int noblock = flags & MSG_DONTWAIT; 9341da177e4SLinus Torvalds struct sock *sk = sock->sk; 9351da177e4SLinus Torvalds struct sk_buff *skb; 9361da177e4SLinus Torvalds int copied, err; 9371da177e4SLinus Torvalds 9381da177e4SLinus Torvalds BT_DBG("sock %p, sk %p", sock, sk); 9391da177e4SLinus Torvalds 9401da177e4SLinus Torvalds if (flags & (MSG_OOB)) 9411da177e4SLinus Torvalds return -EOPNOTSUPP; 9421da177e4SLinus Torvalds 9431da177e4SLinus Torvalds if (sk->sk_state == BT_CLOSED) 9441da177e4SLinus Torvalds return 0; 9451da177e4SLinus Torvalds 94670f23020SAndrei Emeltchenko skb = skb_recv_datagram(sk, flags, noblock, &err); 94770f23020SAndrei Emeltchenko if (!skb) 9481da177e4SLinus Torvalds return err; 9491da177e4SLinus Torvalds 9501da177e4SLinus Torvalds copied = skb->len; 9511da177e4SLinus Torvalds if (len < copied) { 9521da177e4SLinus Torvalds msg->msg_flags |= MSG_TRUNC; 9531da177e4SLinus Torvalds copied = len; 9541da177e4SLinus Torvalds } 9551da177e4SLinus Torvalds 956badff6d0SArnaldo Carvalho de Melo skb_reset_transport_header(skb); 95751f3d02bSDavid S. Miller err = skb_copy_datagram_msg(skb, 0, msg, copied); 9581da177e4SLinus Torvalds 9593a208627SMarcel Holtmann switch (hci_pi(sk)->channel) { 9603a208627SMarcel Holtmann case HCI_CHANNEL_RAW: 9611da177e4SLinus Torvalds hci_sock_cmsg(sk, msg, skb); 9623a208627SMarcel Holtmann break; 96323500189SMarcel Holtmann case HCI_CHANNEL_USER: 964cd82e61cSMarcel Holtmann case HCI_CHANNEL_MONITOR: 965cd82e61cSMarcel Holtmann sock_recv_timestamp(msg, sk, skb); 966cd82e61cSMarcel Holtmann break; 967801c1e8dSJohan Hedberg default: 968801c1e8dSJohan Hedberg if (hci_mgmt_chan_find(hci_pi(sk)->channel)) 969801c1e8dSJohan Hedberg sock_recv_timestamp(msg, sk, skb); 970801c1e8dSJohan Hedberg break; 9713a208627SMarcel Holtmann } 9721da177e4SLinus Torvalds 9731da177e4SLinus Torvalds skb_free_datagram(sk, skb); 9741da177e4SLinus Torvalds 9751da177e4SLinus Torvalds return err ? : copied; 9761da177e4SLinus Torvalds } 9771da177e4SLinus Torvalds 978fa4335d7SJohan Hedberg static int hci_mgmt_cmd(struct hci_mgmt_chan *chan, struct sock *sk, 979fa4335d7SJohan Hedberg struct msghdr *msg, size_t msglen) 980fa4335d7SJohan Hedberg { 981fa4335d7SJohan Hedberg void *buf; 982fa4335d7SJohan Hedberg u8 *cp; 983fa4335d7SJohan Hedberg struct mgmt_hdr *hdr; 984fa4335d7SJohan Hedberg u16 opcode, index, len; 985fa4335d7SJohan Hedberg struct hci_dev *hdev = NULL; 986fa4335d7SJohan Hedberg const struct hci_mgmt_handler *handler; 987fa4335d7SJohan Hedberg bool var_len, no_hdev; 988fa4335d7SJohan Hedberg int err; 989fa4335d7SJohan Hedberg 990fa4335d7SJohan Hedberg BT_DBG("got %zu bytes", msglen); 991fa4335d7SJohan Hedberg 992fa4335d7SJohan Hedberg if (msglen < sizeof(*hdr)) 993fa4335d7SJohan Hedberg return -EINVAL; 994fa4335d7SJohan Hedberg 995fa4335d7SJohan Hedberg buf = kmalloc(msglen, GFP_KERNEL); 996fa4335d7SJohan Hedberg if (!buf) 997fa4335d7SJohan Hedberg return -ENOMEM; 998fa4335d7SJohan Hedberg 999fa4335d7SJohan Hedberg if (memcpy_from_msg(buf, msg, msglen)) { 1000fa4335d7SJohan Hedberg err = -EFAULT; 1001fa4335d7SJohan Hedberg goto done; 1002fa4335d7SJohan Hedberg } 1003fa4335d7SJohan Hedberg 1004fa4335d7SJohan Hedberg hdr = buf; 1005fa4335d7SJohan Hedberg opcode = __le16_to_cpu(hdr->opcode); 1006fa4335d7SJohan Hedberg index = __le16_to_cpu(hdr->index); 1007fa4335d7SJohan Hedberg len = __le16_to_cpu(hdr->len); 1008fa4335d7SJohan Hedberg 1009fa4335d7SJohan Hedberg if (len != msglen - sizeof(*hdr)) { 1010fa4335d7SJohan Hedberg err = -EINVAL; 1011fa4335d7SJohan Hedberg goto done; 1012fa4335d7SJohan Hedberg } 1013fa4335d7SJohan Hedberg 1014fa4335d7SJohan Hedberg if (opcode >= chan->handler_count || 1015fa4335d7SJohan Hedberg chan->handlers[opcode].func == NULL) { 1016fa4335d7SJohan Hedberg BT_DBG("Unknown op %u", opcode); 1017fa4335d7SJohan Hedberg err = mgmt_cmd_status(sk, index, opcode, 1018fa4335d7SJohan Hedberg MGMT_STATUS_UNKNOWN_COMMAND); 1019fa4335d7SJohan Hedberg goto done; 1020fa4335d7SJohan Hedberg } 1021fa4335d7SJohan Hedberg 1022fa4335d7SJohan Hedberg handler = &chan->handlers[opcode]; 1023fa4335d7SJohan Hedberg 1024fa4335d7SJohan Hedberg if (!hci_sock_test_flag(sk, HCI_SOCK_TRUSTED) && 1025fa4335d7SJohan Hedberg !(handler->flags & HCI_MGMT_UNTRUSTED)) { 1026fa4335d7SJohan Hedberg err = mgmt_cmd_status(sk, index, opcode, 1027fa4335d7SJohan Hedberg MGMT_STATUS_PERMISSION_DENIED); 1028fa4335d7SJohan Hedberg goto done; 1029fa4335d7SJohan Hedberg } 1030fa4335d7SJohan Hedberg 1031fa4335d7SJohan Hedberg if (index != MGMT_INDEX_NONE) { 1032fa4335d7SJohan Hedberg hdev = hci_dev_get(index); 1033fa4335d7SJohan Hedberg if (!hdev) { 1034fa4335d7SJohan Hedberg err = mgmt_cmd_status(sk, index, opcode, 1035fa4335d7SJohan Hedberg MGMT_STATUS_INVALID_INDEX); 1036fa4335d7SJohan Hedberg goto done; 1037fa4335d7SJohan Hedberg } 1038fa4335d7SJohan Hedberg 1039fa4335d7SJohan Hedberg if (hci_dev_test_flag(hdev, HCI_SETUP) || 1040fa4335d7SJohan Hedberg hci_dev_test_flag(hdev, HCI_CONFIG) || 1041fa4335d7SJohan Hedberg hci_dev_test_flag(hdev, HCI_USER_CHANNEL)) { 1042fa4335d7SJohan Hedberg err = mgmt_cmd_status(sk, index, opcode, 1043fa4335d7SJohan Hedberg MGMT_STATUS_INVALID_INDEX); 1044fa4335d7SJohan Hedberg goto done; 1045fa4335d7SJohan Hedberg } 1046fa4335d7SJohan Hedberg 1047fa4335d7SJohan Hedberg if (hci_dev_test_flag(hdev, HCI_UNCONFIGURED) && 1048fa4335d7SJohan Hedberg !(handler->flags & HCI_MGMT_UNCONFIGURED)) { 1049fa4335d7SJohan Hedberg err = mgmt_cmd_status(sk, index, opcode, 1050fa4335d7SJohan Hedberg MGMT_STATUS_INVALID_INDEX); 1051fa4335d7SJohan Hedberg goto done; 1052fa4335d7SJohan Hedberg } 1053fa4335d7SJohan Hedberg } 1054fa4335d7SJohan Hedberg 1055fa4335d7SJohan Hedberg no_hdev = (handler->flags & HCI_MGMT_NO_HDEV); 1056fa4335d7SJohan Hedberg if (no_hdev != !hdev) { 1057fa4335d7SJohan Hedberg err = mgmt_cmd_status(sk, index, opcode, 1058fa4335d7SJohan Hedberg MGMT_STATUS_INVALID_INDEX); 1059fa4335d7SJohan Hedberg goto done; 1060fa4335d7SJohan Hedberg } 1061fa4335d7SJohan Hedberg 1062fa4335d7SJohan Hedberg var_len = (handler->flags & HCI_MGMT_VAR_LEN); 1063fa4335d7SJohan Hedberg if ((var_len && len < handler->data_len) || 1064fa4335d7SJohan Hedberg (!var_len && len != handler->data_len)) { 1065fa4335d7SJohan Hedberg err = mgmt_cmd_status(sk, index, opcode, 1066fa4335d7SJohan Hedberg MGMT_STATUS_INVALID_PARAMS); 1067fa4335d7SJohan Hedberg goto done; 1068fa4335d7SJohan Hedberg } 1069fa4335d7SJohan Hedberg 1070fa4335d7SJohan Hedberg if (hdev && chan->hdev_init) 1071fa4335d7SJohan Hedberg chan->hdev_init(sk, hdev); 1072fa4335d7SJohan Hedberg 1073fa4335d7SJohan Hedberg cp = buf + sizeof(*hdr); 1074fa4335d7SJohan Hedberg 1075fa4335d7SJohan Hedberg err = handler->func(sk, hdev, cp, len); 1076fa4335d7SJohan Hedberg if (err < 0) 1077fa4335d7SJohan Hedberg goto done; 1078fa4335d7SJohan Hedberg 1079fa4335d7SJohan Hedberg err = msglen; 1080fa4335d7SJohan Hedberg 1081fa4335d7SJohan Hedberg done: 1082fa4335d7SJohan Hedberg if (hdev) 1083fa4335d7SJohan Hedberg hci_dev_put(hdev); 1084fa4335d7SJohan Hedberg 1085fa4335d7SJohan Hedberg kfree(buf); 1086fa4335d7SJohan Hedberg return err; 1087fa4335d7SJohan Hedberg } 1088fa4335d7SJohan Hedberg 10891b784140SYing Xue static int hci_sock_sendmsg(struct socket *sock, struct msghdr *msg, 10901b784140SYing Xue size_t len) 10911da177e4SLinus Torvalds { 10921da177e4SLinus Torvalds struct sock *sk = sock->sk; 1093801c1e8dSJohan Hedberg struct hci_mgmt_chan *chan; 10941da177e4SLinus Torvalds struct hci_dev *hdev; 10951da177e4SLinus Torvalds struct sk_buff *skb; 10961da177e4SLinus Torvalds int err; 10971da177e4SLinus Torvalds 10981da177e4SLinus Torvalds BT_DBG("sock %p sk %p", sock, sk); 10991da177e4SLinus Torvalds 11001da177e4SLinus Torvalds if (msg->msg_flags & MSG_OOB) 11011da177e4SLinus Torvalds return -EOPNOTSUPP; 11021da177e4SLinus Torvalds 11031da177e4SLinus Torvalds if (msg->msg_flags & ~(MSG_DONTWAIT|MSG_NOSIGNAL|MSG_ERRQUEUE)) 11041da177e4SLinus Torvalds return -EINVAL; 11051da177e4SLinus Torvalds 11061da177e4SLinus Torvalds if (len < 4 || len > HCI_MAX_FRAME_SIZE) 11071da177e4SLinus Torvalds return -EINVAL; 11081da177e4SLinus Torvalds 11091da177e4SLinus Torvalds lock_sock(sk); 11101da177e4SLinus Torvalds 11110381101fSJohan Hedberg switch (hci_pi(sk)->channel) { 11120381101fSJohan Hedberg case HCI_CHANNEL_RAW: 111323500189SMarcel Holtmann case HCI_CHANNEL_USER: 11140381101fSJohan Hedberg break; 1115cd82e61cSMarcel Holtmann case HCI_CHANNEL_MONITOR: 1116cd82e61cSMarcel Holtmann err = -EOPNOTSUPP; 1117cd82e61cSMarcel Holtmann goto done; 11180381101fSJohan Hedberg default: 1119801c1e8dSJohan Hedberg mutex_lock(&mgmt_chan_list_lock); 1120801c1e8dSJohan Hedberg chan = __hci_mgmt_chan_find(hci_pi(sk)->channel); 1121801c1e8dSJohan Hedberg if (chan) 1122fa4335d7SJohan Hedberg err = hci_mgmt_cmd(chan, sk, msg, len); 1123801c1e8dSJohan Hedberg else 11240381101fSJohan Hedberg err = -EINVAL; 1125801c1e8dSJohan Hedberg 1126801c1e8dSJohan Hedberg mutex_unlock(&mgmt_chan_list_lock); 11270381101fSJohan Hedberg goto done; 11280381101fSJohan Hedberg } 11290381101fSJohan Hedberg 113070f23020SAndrei Emeltchenko hdev = hci_pi(sk)->hdev; 113170f23020SAndrei Emeltchenko if (!hdev) { 11321da177e4SLinus Torvalds err = -EBADFD; 11331da177e4SLinus Torvalds goto done; 11341da177e4SLinus Torvalds } 11351da177e4SLinus Torvalds 11367e21addcSMarcel Holtmann if (!test_bit(HCI_UP, &hdev->flags)) { 11377e21addcSMarcel Holtmann err = -ENETDOWN; 11387e21addcSMarcel Holtmann goto done; 11397e21addcSMarcel Holtmann } 11407e21addcSMarcel Holtmann 114170f23020SAndrei Emeltchenko skb = bt_skb_send_alloc(sk, len, msg->msg_flags & MSG_DONTWAIT, &err); 114270f23020SAndrei Emeltchenko if (!skb) 11431da177e4SLinus Torvalds goto done; 11441da177e4SLinus Torvalds 11456ce8e9ceSAl Viro if (memcpy_from_msg(skb_put(skb, len), msg, len)) { 11461da177e4SLinus Torvalds err = -EFAULT; 11471da177e4SLinus Torvalds goto drop; 11481da177e4SLinus Torvalds } 11491da177e4SLinus Torvalds 11500d48d939SMarcel Holtmann bt_cb(skb)->pkt_type = *((unsigned char *) skb->data); 11511da177e4SLinus Torvalds skb_pull(skb, 1); 11521da177e4SLinus Torvalds 11531bc5ad16SMarcel Holtmann if (hci_pi(sk)->channel == HCI_CHANNEL_USER) { 11541bc5ad16SMarcel Holtmann /* No permission check is needed for user channel 11551bc5ad16SMarcel Holtmann * since that gets enforced when binding the socket. 11561bc5ad16SMarcel Holtmann * 11571bc5ad16SMarcel Holtmann * However check that the packet type is valid. 11581bc5ad16SMarcel Holtmann */ 11591bc5ad16SMarcel Holtmann if (bt_cb(skb)->pkt_type != HCI_COMMAND_PKT && 11601bc5ad16SMarcel Holtmann bt_cb(skb)->pkt_type != HCI_ACLDATA_PKT && 11611bc5ad16SMarcel Holtmann bt_cb(skb)->pkt_type != HCI_SCODATA_PKT) { 11621bc5ad16SMarcel Holtmann err = -EINVAL; 11631bc5ad16SMarcel Holtmann goto drop; 11641bc5ad16SMarcel Holtmann } 11651bc5ad16SMarcel Holtmann 11661bc5ad16SMarcel Holtmann skb_queue_tail(&hdev->raw_q, skb); 11671bc5ad16SMarcel Holtmann queue_work(hdev->workqueue, &hdev->tx_work); 11681bc5ad16SMarcel Holtmann } else if (bt_cb(skb)->pkt_type == HCI_COMMAND_PKT) { 116983985319SHarvey Harrison u16 opcode = get_unaligned_le16(skb->data); 11701da177e4SLinus Torvalds u16 ogf = hci_opcode_ogf(opcode); 11711da177e4SLinus Torvalds u16 ocf = hci_opcode_ocf(opcode); 11721da177e4SLinus Torvalds 11731da177e4SLinus Torvalds if (((ogf > HCI_SFLT_MAX_OGF) || 11743bb3c755SGustavo Padovan !hci_test_bit(ocf & HCI_FLT_OCF_BITS, 11753bb3c755SGustavo Padovan &hci_sec_filter.ocf_mask[ogf])) && 11761da177e4SLinus Torvalds !capable(CAP_NET_RAW)) { 11771da177e4SLinus Torvalds err = -EPERM; 11781da177e4SLinus Torvalds goto drop; 11791da177e4SLinus Torvalds } 11801da177e4SLinus Torvalds 1181fee746b0SMarcel Holtmann if (ogf == 0x3f) { 11821da177e4SLinus Torvalds skb_queue_tail(&hdev->raw_q, skb); 11833eff45eaSGustavo F. Padovan queue_work(hdev->workqueue, &hdev->tx_work); 11841da177e4SLinus Torvalds } else { 118549c922bbSStephen Hemminger /* Stand-alone HCI commands must be flagged as 118611714b3dSJohan Hedberg * single-command requests. 118711714b3dSJohan Hedberg */ 1188db6e3e8dSJohan Hedberg bt_cb(skb)->req.start = true; 118911714b3dSJohan Hedberg 11901da177e4SLinus Torvalds skb_queue_tail(&hdev->cmd_q, skb); 1191c347b765SGustavo F. Padovan queue_work(hdev->workqueue, &hdev->cmd_work); 11921da177e4SLinus Torvalds } 11931da177e4SLinus Torvalds } else { 11941da177e4SLinus Torvalds if (!capable(CAP_NET_RAW)) { 11951da177e4SLinus Torvalds err = -EPERM; 11961da177e4SLinus Torvalds goto drop; 11971da177e4SLinus Torvalds } 11981da177e4SLinus Torvalds 11991da177e4SLinus Torvalds skb_queue_tail(&hdev->raw_q, skb); 12003eff45eaSGustavo F. Padovan queue_work(hdev->workqueue, &hdev->tx_work); 12011da177e4SLinus Torvalds } 12021da177e4SLinus Torvalds 12031da177e4SLinus Torvalds err = len; 12041da177e4SLinus Torvalds 12051da177e4SLinus Torvalds done: 12061da177e4SLinus Torvalds release_sock(sk); 12071da177e4SLinus Torvalds return err; 12081da177e4SLinus Torvalds 12091da177e4SLinus Torvalds drop: 12101da177e4SLinus Torvalds kfree_skb(skb); 12111da177e4SLinus Torvalds goto done; 12121da177e4SLinus Torvalds } 12131da177e4SLinus Torvalds 12148fc9ced3SGustavo Padovan static int hci_sock_setsockopt(struct socket *sock, int level, int optname, 12158fc9ced3SGustavo Padovan char __user *optval, unsigned int len) 12161da177e4SLinus Torvalds { 12171da177e4SLinus Torvalds struct hci_ufilter uf = { .opcode = 0 }; 12181da177e4SLinus Torvalds struct sock *sk = sock->sk; 12191da177e4SLinus Torvalds int err = 0, opt = 0; 12201da177e4SLinus Torvalds 12211da177e4SLinus Torvalds BT_DBG("sk %p, opt %d", sk, optname); 12221da177e4SLinus Torvalds 12231da177e4SLinus Torvalds lock_sock(sk); 12241da177e4SLinus Torvalds 12252f39cdb7SMarcel Holtmann if (hci_pi(sk)->channel != HCI_CHANNEL_RAW) { 1226c2371e80SMarcel Holtmann err = -EBADFD; 12272f39cdb7SMarcel Holtmann goto done; 12282f39cdb7SMarcel Holtmann } 12292f39cdb7SMarcel Holtmann 12301da177e4SLinus Torvalds switch (optname) { 12311da177e4SLinus Torvalds case HCI_DATA_DIR: 12321da177e4SLinus Torvalds if (get_user(opt, (int __user *)optval)) { 12331da177e4SLinus Torvalds err = -EFAULT; 12341da177e4SLinus Torvalds break; 12351da177e4SLinus Torvalds } 12361da177e4SLinus Torvalds 12371da177e4SLinus Torvalds if (opt) 12381da177e4SLinus Torvalds hci_pi(sk)->cmsg_mask |= HCI_CMSG_DIR; 12391da177e4SLinus Torvalds else 12401da177e4SLinus Torvalds hci_pi(sk)->cmsg_mask &= ~HCI_CMSG_DIR; 12411da177e4SLinus Torvalds break; 12421da177e4SLinus Torvalds 12431da177e4SLinus Torvalds case HCI_TIME_STAMP: 12441da177e4SLinus Torvalds if (get_user(opt, (int __user *)optval)) { 12451da177e4SLinus Torvalds err = -EFAULT; 12461da177e4SLinus Torvalds break; 12471da177e4SLinus Torvalds } 12481da177e4SLinus Torvalds 12491da177e4SLinus Torvalds if (opt) 12501da177e4SLinus Torvalds hci_pi(sk)->cmsg_mask |= HCI_CMSG_TSTAMP; 12511da177e4SLinus Torvalds else 12521da177e4SLinus Torvalds hci_pi(sk)->cmsg_mask &= ~HCI_CMSG_TSTAMP; 12531da177e4SLinus Torvalds break; 12541da177e4SLinus Torvalds 12551da177e4SLinus Torvalds case HCI_FILTER: 12560878b666SMarcel Holtmann { 12570878b666SMarcel Holtmann struct hci_filter *f = &hci_pi(sk)->filter; 12580878b666SMarcel Holtmann 12590878b666SMarcel Holtmann uf.type_mask = f->type_mask; 12600878b666SMarcel Holtmann uf.opcode = f->opcode; 12610878b666SMarcel Holtmann uf.event_mask[0] = *((u32 *) f->event_mask + 0); 12620878b666SMarcel Holtmann uf.event_mask[1] = *((u32 *) f->event_mask + 1); 12630878b666SMarcel Holtmann } 12640878b666SMarcel Holtmann 12651da177e4SLinus Torvalds len = min_t(unsigned int, len, sizeof(uf)); 12661da177e4SLinus Torvalds if (copy_from_user(&uf, optval, len)) { 12671da177e4SLinus Torvalds err = -EFAULT; 12681da177e4SLinus Torvalds break; 12691da177e4SLinus Torvalds } 12701da177e4SLinus Torvalds 12711da177e4SLinus Torvalds if (!capable(CAP_NET_RAW)) { 12721da177e4SLinus Torvalds uf.type_mask &= hci_sec_filter.type_mask; 12731da177e4SLinus Torvalds uf.event_mask[0] &= *((u32 *) hci_sec_filter.event_mask + 0); 12741da177e4SLinus Torvalds uf.event_mask[1] &= *((u32 *) hci_sec_filter.event_mask + 1); 12751da177e4SLinus Torvalds } 12761da177e4SLinus Torvalds 12771da177e4SLinus Torvalds { 12781da177e4SLinus Torvalds struct hci_filter *f = &hci_pi(sk)->filter; 12791da177e4SLinus Torvalds 12801da177e4SLinus Torvalds f->type_mask = uf.type_mask; 12811da177e4SLinus Torvalds f->opcode = uf.opcode; 12821da177e4SLinus Torvalds *((u32 *) f->event_mask + 0) = uf.event_mask[0]; 12831da177e4SLinus Torvalds *((u32 *) f->event_mask + 1) = uf.event_mask[1]; 12841da177e4SLinus Torvalds } 12851da177e4SLinus Torvalds break; 12861da177e4SLinus Torvalds 12871da177e4SLinus Torvalds default: 12881da177e4SLinus Torvalds err = -ENOPROTOOPT; 12891da177e4SLinus Torvalds break; 12901da177e4SLinus Torvalds } 12911da177e4SLinus Torvalds 12922f39cdb7SMarcel Holtmann done: 12931da177e4SLinus Torvalds release_sock(sk); 12941da177e4SLinus Torvalds return err; 12951da177e4SLinus Torvalds } 12961da177e4SLinus Torvalds 12978fc9ced3SGustavo Padovan static int hci_sock_getsockopt(struct socket *sock, int level, int optname, 12988fc9ced3SGustavo Padovan char __user *optval, int __user *optlen) 12991da177e4SLinus Torvalds { 13001da177e4SLinus Torvalds struct hci_ufilter uf; 13011da177e4SLinus Torvalds struct sock *sk = sock->sk; 1302cedc5469SMarcel Holtmann int len, opt, err = 0; 1303cedc5469SMarcel Holtmann 1304cedc5469SMarcel Holtmann BT_DBG("sk %p, opt %d", sk, optname); 13051da177e4SLinus Torvalds 13061da177e4SLinus Torvalds if (get_user(len, optlen)) 13071da177e4SLinus Torvalds return -EFAULT; 13081da177e4SLinus Torvalds 1309cedc5469SMarcel Holtmann lock_sock(sk); 1310cedc5469SMarcel Holtmann 1311cedc5469SMarcel Holtmann if (hci_pi(sk)->channel != HCI_CHANNEL_RAW) { 1312c2371e80SMarcel Holtmann err = -EBADFD; 1313cedc5469SMarcel Holtmann goto done; 1314cedc5469SMarcel Holtmann } 1315cedc5469SMarcel Holtmann 13161da177e4SLinus Torvalds switch (optname) { 13171da177e4SLinus Torvalds case HCI_DATA_DIR: 13181da177e4SLinus Torvalds if (hci_pi(sk)->cmsg_mask & HCI_CMSG_DIR) 13191da177e4SLinus Torvalds opt = 1; 13201da177e4SLinus Torvalds else 13211da177e4SLinus Torvalds opt = 0; 13221da177e4SLinus Torvalds 13231da177e4SLinus Torvalds if (put_user(opt, optval)) 1324cedc5469SMarcel Holtmann err = -EFAULT; 13251da177e4SLinus Torvalds break; 13261da177e4SLinus Torvalds 13271da177e4SLinus Torvalds case HCI_TIME_STAMP: 13281da177e4SLinus Torvalds if (hci_pi(sk)->cmsg_mask & HCI_CMSG_TSTAMP) 13291da177e4SLinus Torvalds opt = 1; 13301da177e4SLinus Torvalds else 13311da177e4SLinus Torvalds opt = 0; 13321da177e4SLinus Torvalds 13331da177e4SLinus Torvalds if (put_user(opt, optval)) 1334cedc5469SMarcel Holtmann err = -EFAULT; 13351da177e4SLinus Torvalds break; 13361da177e4SLinus Torvalds 13371da177e4SLinus Torvalds case HCI_FILTER: 13381da177e4SLinus Torvalds { 13391da177e4SLinus Torvalds struct hci_filter *f = &hci_pi(sk)->filter; 13401da177e4SLinus Torvalds 1341e15ca9a0SMathias Krause memset(&uf, 0, sizeof(uf)); 13421da177e4SLinus Torvalds uf.type_mask = f->type_mask; 13431da177e4SLinus Torvalds uf.opcode = f->opcode; 13441da177e4SLinus Torvalds uf.event_mask[0] = *((u32 *) f->event_mask + 0); 13451da177e4SLinus Torvalds uf.event_mask[1] = *((u32 *) f->event_mask + 1); 13461da177e4SLinus Torvalds } 13471da177e4SLinus Torvalds 13481da177e4SLinus Torvalds len = min_t(unsigned int, len, sizeof(uf)); 13491da177e4SLinus Torvalds if (copy_to_user(optval, &uf, len)) 1350cedc5469SMarcel Holtmann err = -EFAULT; 13511da177e4SLinus Torvalds break; 13521da177e4SLinus Torvalds 13531da177e4SLinus Torvalds default: 1354cedc5469SMarcel Holtmann err = -ENOPROTOOPT; 13551da177e4SLinus Torvalds break; 13561da177e4SLinus Torvalds } 13571da177e4SLinus Torvalds 1358cedc5469SMarcel Holtmann done: 1359cedc5469SMarcel Holtmann release_sock(sk); 1360cedc5469SMarcel Holtmann return err; 13611da177e4SLinus Torvalds } 13621da177e4SLinus Torvalds 136390ddc4f0SEric Dumazet static const struct proto_ops hci_sock_ops = { 13641da177e4SLinus Torvalds .family = PF_BLUETOOTH, 13651da177e4SLinus Torvalds .owner = THIS_MODULE, 13661da177e4SLinus Torvalds .release = hci_sock_release, 13671da177e4SLinus Torvalds .bind = hci_sock_bind, 13681da177e4SLinus Torvalds .getname = hci_sock_getname, 13691da177e4SLinus Torvalds .sendmsg = hci_sock_sendmsg, 13701da177e4SLinus Torvalds .recvmsg = hci_sock_recvmsg, 13711da177e4SLinus Torvalds .ioctl = hci_sock_ioctl, 13721da177e4SLinus Torvalds .poll = datagram_poll, 13731da177e4SLinus Torvalds .listen = sock_no_listen, 13741da177e4SLinus Torvalds .shutdown = sock_no_shutdown, 13751da177e4SLinus Torvalds .setsockopt = hci_sock_setsockopt, 13761da177e4SLinus Torvalds .getsockopt = hci_sock_getsockopt, 13771da177e4SLinus Torvalds .connect = sock_no_connect, 13781da177e4SLinus Torvalds .socketpair = sock_no_socketpair, 13791da177e4SLinus Torvalds .accept = sock_no_accept, 13801da177e4SLinus Torvalds .mmap = sock_no_mmap 13811da177e4SLinus Torvalds }; 13821da177e4SLinus Torvalds 13831da177e4SLinus Torvalds static struct proto hci_sk_proto = { 13841da177e4SLinus Torvalds .name = "HCI", 13851da177e4SLinus Torvalds .owner = THIS_MODULE, 13861da177e4SLinus Torvalds .obj_size = sizeof(struct hci_pinfo) 13871da177e4SLinus Torvalds }; 13881da177e4SLinus Torvalds 13893f378b68SEric Paris static int hci_sock_create(struct net *net, struct socket *sock, int protocol, 13903f378b68SEric Paris int kern) 13911da177e4SLinus Torvalds { 13921da177e4SLinus Torvalds struct sock *sk; 13931da177e4SLinus Torvalds 13941da177e4SLinus Torvalds BT_DBG("sock %p", sock); 13951da177e4SLinus Torvalds 13961da177e4SLinus Torvalds if (sock->type != SOCK_RAW) 13971da177e4SLinus Torvalds return -ESOCKTNOSUPPORT; 13981da177e4SLinus Torvalds 13991da177e4SLinus Torvalds sock->ops = &hci_sock_ops; 14001da177e4SLinus Torvalds 140111aa9c28SEric W. Biederman sk = sk_alloc(net, PF_BLUETOOTH, GFP_ATOMIC, &hci_sk_proto, kern); 14021da177e4SLinus Torvalds if (!sk) 14031da177e4SLinus Torvalds return -ENOMEM; 14041da177e4SLinus Torvalds 14051da177e4SLinus Torvalds sock_init_data(sock, sk); 14061da177e4SLinus Torvalds 14071da177e4SLinus Torvalds sock_reset_flag(sk, SOCK_ZAPPED); 14081da177e4SLinus Torvalds 14091da177e4SLinus Torvalds sk->sk_protocol = protocol; 14101da177e4SLinus Torvalds 14111da177e4SLinus Torvalds sock->state = SS_UNCONNECTED; 14121da177e4SLinus Torvalds sk->sk_state = BT_OPEN; 14131da177e4SLinus Torvalds 14141da177e4SLinus Torvalds bt_sock_link(&hci_sk_list, sk); 14151da177e4SLinus Torvalds return 0; 14161da177e4SLinus Torvalds } 14171da177e4SLinus Torvalds 1418ec1b4cf7SStephen Hemminger static const struct net_proto_family hci_sock_family_ops = { 14191da177e4SLinus Torvalds .family = PF_BLUETOOTH, 14201da177e4SLinus Torvalds .owner = THIS_MODULE, 14211da177e4SLinus Torvalds .create = hci_sock_create, 14221da177e4SLinus Torvalds }; 14231da177e4SLinus Torvalds 14241da177e4SLinus Torvalds int __init hci_sock_init(void) 14251da177e4SLinus Torvalds { 14261da177e4SLinus Torvalds int err; 14271da177e4SLinus Torvalds 1428b0a8e282SMarcel Holtmann BUILD_BUG_ON(sizeof(struct sockaddr_hci) > sizeof(struct sockaddr)); 1429b0a8e282SMarcel Holtmann 14301da177e4SLinus Torvalds err = proto_register(&hci_sk_proto, 0); 14311da177e4SLinus Torvalds if (err < 0) 14321da177e4SLinus Torvalds return err; 14331da177e4SLinus Torvalds 14341da177e4SLinus Torvalds err = bt_sock_register(BTPROTO_HCI, &hci_sock_family_ops); 1435f7c86637SMasatake YAMATO if (err < 0) { 1436f7c86637SMasatake YAMATO BT_ERR("HCI socket registration failed"); 14371da177e4SLinus Torvalds goto error; 1438f7c86637SMasatake YAMATO } 1439f7c86637SMasatake YAMATO 1440b0316615SAl Viro err = bt_procfs_init(&init_net, "hci", &hci_sk_list, NULL); 1441f7c86637SMasatake YAMATO if (err < 0) { 1442f7c86637SMasatake YAMATO BT_ERR("Failed to create HCI proc file"); 1443f7c86637SMasatake YAMATO bt_sock_unregister(BTPROTO_HCI); 1444f7c86637SMasatake YAMATO goto error; 1445f7c86637SMasatake YAMATO } 14461da177e4SLinus Torvalds 14471da177e4SLinus Torvalds BT_INFO("HCI socket layer initialized"); 14481da177e4SLinus Torvalds 14491da177e4SLinus Torvalds return 0; 14501da177e4SLinus Torvalds 14511da177e4SLinus Torvalds error: 14521da177e4SLinus Torvalds proto_unregister(&hci_sk_proto); 14531da177e4SLinus Torvalds return err; 14541da177e4SLinus Torvalds } 14551da177e4SLinus Torvalds 1456b7440a14SAnand Gadiyar void hci_sock_cleanup(void) 14571da177e4SLinus Torvalds { 1458f7c86637SMasatake YAMATO bt_procfs_cleanup(&init_net, "hci"); 14595e9d7f86SDavid Herrmann bt_sock_unregister(BTPROTO_HCI); 14601da177e4SLinus Torvalds proto_unregister(&hci_sk_proto); 14611da177e4SLinus Torvalds } 1462