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> 28787b306cSJohannes Berg #include <linux/utsname.h> 2970ecce91SMarcel Holtmann #include <linux/sched.h> 301da177e4SLinus Torvalds #include <asm/unaligned.h> 311da177e4SLinus Torvalds 321da177e4SLinus Torvalds #include <net/bluetooth/bluetooth.h> 331da177e4SLinus Torvalds #include <net/bluetooth/hci_core.h> 34cd82e61cSMarcel Holtmann #include <net/bluetooth/hci_mon.h> 35fa4335d7SJohan Hedberg #include <net/bluetooth/mgmt.h> 36fa4335d7SJohan Hedberg 37fa4335d7SJohan Hedberg #include "mgmt_util.h" 381da177e4SLinus Torvalds 39801c1e8dSJohan Hedberg static LIST_HEAD(mgmt_chan_list); 40801c1e8dSJohan Hedberg static DEFINE_MUTEX(mgmt_chan_list_lock); 41801c1e8dSJohan Hedberg 4270ecce91SMarcel Holtmann static DEFINE_IDA(sock_cookie_ida); 4370ecce91SMarcel Holtmann 44cd82e61cSMarcel Holtmann static atomic_t monitor_promisc = ATOMIC_INIT(0); 45cd82e61cSMarcel Holtmann 461da177e4SLinus Torvalds /* ----- HCI socket interface ----- */ 471da177e4SLinus Torvalds 48863def58SMarcel Holtmann /* Socket info */ 49863def58SMarcel Holtmann #define hci_pi(sk) ((struct hci_pinfo *) sk) 50863def58SMarcel Holtmann 51863def58SMarcel Holtmann struct hci_pinfo { 52863def58SMarcel Holtmann struct bt_sock bt; 53863def58SMarcel Holtmann struct hci_dev *hdev; 54863def58SMarcel Holtmann struct hci_filter filter; 55863def58SMarcel Holtmann __u32 cmsg_mask; 56863def58SMarcel Holtmann unsigned short channel; 576befc644SMarcel Holtmann unsigned long flags; 5870ecce91SMarcel Holtmann __u32 cookie; 5970ecce91SMarcel Holtmann char comm[TASK_COMM_LEN]; 60863def58SMarcel Holtmann }; 61863def58SMarcel Holtmann 626befc644SMarcel Holtmann void hci_sock_set_flag(struct sock *sk, int nr) 636befc644SMarcel Holtmann { 646befc644SMarcel Holtmann set_bit(nr, &hci_pi(sk)->flags); 656befc644SMarcel Holtmann } 666befc644SMarcel Holtmann 676befc644SMarcel Holtmann void hci_sock_clear_flag(struct sock *sk, int nr) 686befc644SMarcel Holtmann { 696befc644SMarcel Holtmann clear_bit(nr, &hci_pi(sk)->flags); 706befc644SMarcel Holtmann } 716befc644SMarcel Holtmann 72c85be545SMarcel Holtmann int hci_sock_test_flag(struct sock *sk, int nr) 73c85be545SMarcel Holtmann { 74c85be545SMarcel Holtmann return test_bit(nr, &hci_pi(sk)->flags); 75c85be545SMarcel Holtmann } 76c85be545SMarcel Holtmann 77d0f172b1SJohan Hedberg unsigned short hci_sock_get_channel(struct sock *sk) 78d0f172b1SJohan Hedberg { 79d0f172b1SJohan Hedberg return hci_pi(sk)->channel; 80d0f172b1SJohan Hedberg } 81d0f172b1SJohan Hedberg 8270ecce91SMarcel Holtmann u32 hci_sock_get_cookie(struct sock *sk) 8370ecce91SMarcel Holtmann { 8470ecce91SMarcel Holtmann return hci_pi(sk)->cookie; 8570ecce91SMarcel Holtmann } 8670ecce91SMarcel Holtmann 879391976aSJiri Slaby static inline int hci_test_bit(int nr, const void *addr) 881da177e4SLinus Torvalds { 899391976aSJiri Slaby return *((const __u32 *) addr + (nr >> 5)) & ((__u32) 1 << (nr & 31)); 901da177e4SLinus Torvalds } 911da177e4SLinus Torvalds 921da177e4SLinus Torvalds /* Security filter */ 933ad254f7SMarcel Holtmann #define HCI_SFLT_MAX_OGF 5 943ad254f7SMarcel Holtmann 953ad254f7SMarcel Holtmann struct hci_sec_filter { 963ad254f7SMarcel Holtmann __u32 type_mask; 973ad254f7SMarcel Holtmann __u32 event_mask[2]; 983ad254f7SMarcel Holtmann __u32 ocf_mask[HCI_SFLT_MAX_OGF + 1][4]; 993ad254f7SMarcel Holtmann }; 1003ad254f7SMarcel Holtmann 1017e67c112SMarcel Holtmann static const struct hci_sec_filter hci_sec_filter = { 1021da177e4SLinus Torvalds /* Packet types */ 1031da177e4SLinus Torvalds 0x10, 1041da177e4SLinus Torvalds /* Events */ 105dd7f5527SMarcel Holtmann { 0x1000d9fe, 0x0000b00c }, 1061da177e4SLinus Torvalds /* Commands */ 1071da177e4SLinus Torvalds { 1081da177e4SLinus Torvalds { 0x0 }, 1091da177e4SLinus Torvalds /* OGF_LINK_CTL */ 1107c631a67SMarcel Holtmann { 0xbe000006, 0x00000001, 0x00000000, 0x00 }, 1111da177e4SLinus Torvalds /* OGF_LINK_POLICY */ 1127c631a67SMarcel Holtmann { 0x00005200, 0x00000000, 0x00000000, 0x00 }, 1131da177e4SLinus Torvalds /* OGF_HOST_CTL */ 1147c631a67SMarcel Holtmann { 0xaab00200, 0x2b402aaa, 0x05220154, 0x00 }, 1151da177e4SLinus Torvalds /* OGF_INFO_PARAM */ 1167c631a67SMarcel Holtmann { 0x000002be, 0x00000000, 0x00000000, 0x00 }, 1171da177e4SLinus Torvalds /* OGF_STATUS_PARAM */ 1187c631a67SMarcel Holtmann { 0x000000ea, 0x00000000, 0x00000000, 0x00 } 1191da177e4SLinus Torvalds } 1201da177e4SLinus Torvalds }; 1211da177e4SLinus Torvalds 1221da177e4SLinus Torvalds static struct bt_sock_list hci_sk_list = { 123d5fb2962SRobert P. J. Day .lock = __RW_LOCK_UNLOCKED(hci_sk_list.lock) 1241da177e4SLinus Torvalds }; 1251da177e4SLinus Torvalds 126f81fe64fSMarcel Holtmann static bool is_filtered_packet(struct sock *sk, struct sk_buff *skb) 127f81fe64fSMarcel Holtmann { 128f81fe64fSMarcel Holtmann struct hci_filter *flt; 129f81fe64fSMarcel Holtmann int flt_type, flt_event; 130f81fe64fSMarcel Holtmann 131f81fe64fSMarcel Holtmann /* Apply filter */ 132f81fe64fSMarcel Holtmann flt = &hci_pi(sk)->filter; 133f81fe64fSMarcel Holtmann 134d79f34e3SMarcel Holtmann flt_type = hci_skb_pkt_type(skb) & HCI_FLT_TYPE_BITS; 135f81fe64fSMarcel Holtmann 136f81fe64fSMarcel Holtmann if (!test_bit(flt_type, &flt->type_mask)) 137f81fe64fSMarcel Holtmann return true; 138f81fe64fSMarcel Holtmann 139f81fe64fSMarcel Holtmann /* Extra filter for event packets only */ 140d79f34e3SMarcel Holtmann if (hci_skb_pkt_type(skb) != HCI_EVENT_PKT) 141f81fe64fSMarcel Holtmann return false; 142f81fe64fSMarcel Holtmann 143f81fe64fSMarcel Holtmann flt_event = (*(__u8 *)skb->data & HCI_FLT_EVENT_BITS); 144f81fe64fSMarcel Holtmann 145f81fe64fSMarcel Holtmann if (!hci_test_bit(flt_event, &flt->event_mask)) 146f81fe64fSMarcel Holtmann return true; 147f81fe64fSMarcel Holtmann 148f81fe64fSMarcel Holtmann /* Check filter only when opcode is set */ 149f81fe64fSMarcel Holtmann if (!flt->opcode) 150f81fe64fSMarcel Holtmann return false; 151f81fe64fSMarcel Holtmann 152f81fe64fSMarcel Holtmann if (flt_event == HCI_EV_CMD_COMPLETE && 153f81fe64fSMarcel Holtmann flt->opcode != get_unaligned((__le16 *)(skb->data + 3))) 154f81fe64fSMarcel Holtmann return true; 155f81fe64fSMarcel Holtmann 156f81fe64fSMarcel Holtmann if (flt_event == HCI_EV_CMD_STATUS && 157f81fe64fSMarcel Holtmann flt->opcode != get_unaligned((__le16 *)(skb->data + 4))) 158f81fe64fSMarcel Holtmann return true; 159f81fe64fSMarcel Holtmann 160f81fe64fSMarcel Holtmann return false; 161f81fe64fSMarcel Holtmann } 162f81fe64fSMarcel Holtmann 1631da177e4SLinus Torvalds /* Send frame to RAW socket */ 164470fe1b5SMarcel Holtmann void hci_send_to_sock(struct hci_dev *hdev, struct sk_buff *skb) 1651da177e4SLinus Torvalds { 1661da177e4SLinus Torvalds struct sock *sk; 167e0edf373SMarcel Holtmann struct sk_buff *skb_copy = NULL; 1681da177e4SLinus Torvalds 1691da177e4SLinus Torvalds BT_DBG("hdev %p len %d", hdev, skb->len); 1701da177e4SLinus Torvalds 1711da177e4SLinus Torvalds read_lock(&hci_sk_list.lock); 172470fe1b5SMarcel Holtmann 173b67bfe0dSSasha Levin sk_for_each(sk, &hci_sk_list.head) { 1741da177e4SLinus Torvalds struct sk_buff *nskb; 1751da177e4SLinus Torvalds 1761da177e4SLinus Torvalds if (sk->sk_state != BT_BOUND || hci_pi(sk)->hdev != hdev) 1771da177e4SLinus Torvalds continue; 1781da177e4SLinus Torvalds 1791da177e4SLinus Torvalds /* Don't send frame to the socket it came from */ 1801da177e4SLinus Torvalds if (skb->sk == sk) 1811da177e4SLinus Torvalds continue; 1821da177e4SLinus Torvalds 18323500189SMarcel Holtmann if (hci_pi(sk)->channel == HCI_CHANNEL_RAW) { 184d79f34e3SMarcel Holtmann if (hci_skb_pkt_type(skb) != HCI_COMMAND_PKT && 185d79f34e3SMarcel Holtmann hci_skb_pkt_type(skb) != HCI_EVENT_PKT && 186d79f34e3SMarcel Holtmann hci_skb_pkt_type(skb) != HCI_ACLDATA_PKT && 187d79f34e3SMarcel Holtmann hci_skb_pkt_type(skb) != HCI_SCODATA_PKT) 188bb77543eSMarcel Holtmann continue; 189f81fe64fSMarcel Holtmann if (is_filtered_packet(sk, skb)) 1901da177e4SLinus Torvalds continue; 19123500189SMarcel Holtmann } else if (hci_pi(sk)->channel == HCI_CHANNEL_USER) { 19223500189SMarcel Holtmann if (!bt_cb(skb)->incoming) 19323500189SMarcel Holtmann continue; 194d79f34e3SMarcel Holtmann if (hci_skb_pkt_type(skb) != HCI_EVENT_PKT && 195d79f34e3SMarcel Holtmann hci_skb_pkt_type(skb) != HCI_ACLDATA_PKT && 196d79f34e3SMarcel Holtmann hci_skb_pkt_type(skb) != HCI_SCODATA_PKT) 19723500189SMarcel Holtmann continue; 19823500189SMarcel Holtmann } else { 19923500189SMarcel Holtmann /* Don't send frame to other channel types */ 20023500189SMarcel Holtmann continue; 20123500189SMarcel Holtmann } 2021da177e4SLinus Torvalds 203e0edf373SMarcel Holtmann if (!skb_copy) { 204e0edf373SMarcel Holtmann /* Create a private copy with headroom */ 205bad93e9dSOctavian Purdila skb_copy = __pskb_copy_fclone(skb, 1, GFP_ATOMIC, true); 206e0edf373SMarcel Holtmann if (!skb_copy) 2071da177e4SLinus Torvalds continue; 2081da177e4SLinus Torvalds 2091da177e4SLinus Torvalds /* Put type byte before the data */ 210d79f34e3SMarcel Holtmann memcpy(skb_push(skb_copy, 1), &hci_skb_pkt_type(skb), 1); 211e0edf373SMarcel Holtmann } 212e0edf373SMarcel Holtmann 213e0edf373SMarcel Holtmann nskb = skb_clone(skb_copy, GFP_ATOMIC); 214e0edf373SMarcel Holtmann if (!nskb) 215e0edf373SMarcel Holtmann continue; 2161da177e4SLinus Torvalds 2171da177e4SLinus Torvalds if (sock_queue_rcv_skb(sk, nskb)) 2181da177e4SLinus Torvalds kfree_skb(nskb); 2191da177e4SLinus Torvalds } 220470fe1b5SMarcel Holtmann 221470fe1b5SMarcel Holtmann read_unlock(&hci_sk_list.lock); 222e0edf373SMarcel Holtmann 223e0edf373SMarcel Holtmann kfree_skb(skb_copy); 224470fe1b5SMarcel Holtmann } 225470fe1b5SMarcel Holtmann 2267129069eSJohan Hedberg /* Send frame to sockets with specific channel */ 2277129069eSJohan Hedberg void hci_send_to_channel(unsigned short channel, struct sk_buff *skb, 228c08b1a1dSMarcel Holtmann int flag, struct sock *skip_sk) 229470fe1b5SMarcel Holtmann { 230470fe1b5SMarcel Holtmann struct sock *sk; 231470fe1b5SMarcel Holtmann 2327129069eSJohan Hedberg BT_DBG("channel %u len %d", channel, skb->len); 233470fe1b5SMarcel Holtmann 234470fe1b5SMarcel Holtmann read_lock(&hci_sk_list.lock); 235470fe1b5SMarcel Holtmann 236b67bfe0dSSasha Levin sk_for_each(sk, &hci_sk_list.head) { 237470fe1b5SMarcel Holtmann struct sk_buff *nskb; 238470fe1b5SMarcel Holtmann 239c08b1a1dSMarcel Holtmann /* Ignore socket without the flag set */ 240c85be545SMarcel Holtmann if (!hci_sock_test_flag(sk, flag)) 241c08b1a1dSMarcel Holtmann continue; 242c08b1a1dSMarcel Holtmann 243470fe1b5SMarcel Holtmann /* Skip the original socket */ 244470fe1b5SMarcel Holtmann if (sk == skip_sk) 245470fe1b5SMarcel Holtmann continue; 246470fe1b5SMarcel Holtmann 247470fe1b5SMarcel Holtmann if (sk->sk_state != BT_BOUND) 248470fe1b5SMarcel Holtmann continue; 249470fe1b5SMarcel Holtmann 2507129069eSJohan Hedberg if (hci_pi(sk)->channel != channel) 251d7f72f61SMarcel Holtmann continue; 252d7f72f61SMarcel Holtmann 253d7f72f61SMarcel Holtmann nskb = skb_clone(skb, GFP_ATOMIC); 254d7f72f61SMarcel Holtmann if (!nskb) 255d7f72f61SMarcel Holtmann continue; 256d7f72f61SMarcel Holtmann 257d7f72f61SMarcel Holtmann if (sock_queue_rcv_skb(sk, nskb)) 258d7f72f61SMarcel Holtmann kfree_skb(nskb); 259d7f72f61SMarcel Holtmann } 260d7f72f61SMarcel Holtmann 261d7f72f61SMarcel Holtmann read_unlock(&hci_sk_list.lock); 262d7f72f61SMarcel Holtmann } 263d7f72f61SMarcel Holtmann 264cd82e61cSMarcel Holtmann /* Send frame to monitor socket */ 265cd82e61cSMarcel Holtmann void hci_send_to_monitor(struct hci_dev *hdev, struct sk_buff *skb) 266cd82e61cSMarcel Holtmann { 267cd82e61cSMarcel Holtmann struct sk_buff *skb_copy = NULL; 2682b531294SMarcel Holtmann struct hci_mon_hdr *hdr; 269cd82e61cSMarcel Holtmann __le16 opcode; 270cd82e61cSMarcel Holtmann 271cd82e61cSMarcel Holtmann if (!atomic_read(&monitor_promisc)) 272cd82e61cSMarcel Holtmann return; 273cd82e61cSMarcel Holtmann 274cd82e61cSMarcel Holtmann BT_DBG("hdev %p len %d", hdev, skb->len); 275cd82e61cSMarcel Holtmann 276d79f34e3SMarcel Holtmann switch (hci_skb_pkt_type(skb)) { 277cd82e61cSMarcel Holtmann case HCI_COMMAND_PKT: 278dcf4adbfSJoe Perches opcode = cpu_to_le16(HCI_MON_COMMAND_PKT); 279cd82e61cSMarcel Holtmann break; 280cd82e61cSMarcel Holtmann case HCI_EVENT_PKT: 281dcf4adbfSJoe Perches opcode = cpu_to_le16(HCI_MON_EVENT_PKT); 282cd82e61cSMarcel Holtmann break; 283cd82e61cSMarcel Holtmann case HCI_ACLDATA_PKT: 284cd82e61cSMarcel Holtmann if (bt_cb(skb)->incoming) 285dcf4adbfSJoe Perches opcode = cpu_to_le16(HCI_MON_ACL_RX_PKT); 286cd82e61cSMarcel Holtmann else 287dcf4adbfSJoe Perches opcode = cpu_to_le16(HCI_MON_ACL_TX_PKT); 288cd82e61cSMarcel Holtmann break; 289cd82e61cSMarcel Holtmann case HCI_SCODATA_PKT: 290cd82e61cSMarcel Holtmann if (bt_cb(skb)->incoming) 291dcf4adbfSJoe Perches opcode = cpu_to_le16(HCI_MON_SCO_RX_PKT); 292cd82e61cSMarcel Holtmann else 293dcf4adbfSJoe Perches opcode = cpu_to_le16(HCI_MON_SCO_TX_PKT); 294cd82e61cSMarcel Holtmann break; 295e875ff84SMarcel Holtmann case HCI_DIAG_PKT: 296e875ff84SMarcel Holtmann opcode = cpu_to_le16(HCI_MON_VENDOR_DIAG); 297e875ff84SMarcel Holtmann break; 298cd82e61cSMarcel Holtmann default: 299cd82e61cSMarcel Holtmann return; 300cd82e61cSMarcel Holtmann } 301cd82e61cSMarcel Holtmann 3022b531294SMarcel Holtmann /* Create a private copy with headroom */ 3032b531294SMarcel Holtmann skb_copy = __pskb_copy_fclone(skb, HCI_MON_HDR_SIZE, GFP_ATOMIC, true); 3042b531294SMarcel Holtmann if (!skb_copy) 3052b531294SMarcel Holtmann return; 3062b531294SMarcel Holtmann 3072b531294SMarcel Holtmann /* Put header before the data */ 3082b531294SMarcel Holtmann hdr = (void *)skb_push(skb_copy, HCI_MON_HDR_SIZE); 3092b531294SMarcel Holtmann hdr->opcode = opcode; 3102b531294SMarcel Holtmann hdr->index = cpu_to_le16(hdev->id); 3112b531294SMarcel Holtmann hdr->len = cpu_to_le16(skb->len); 3122b531294SMarcel Holtmann 313c08b1a1dSMarcel Holtmann hci_send_to_channel(HCI_CHANNEL_MONITOR, skb_copy, 314c08b1a1dSMarcel Holtmann HCI_SOCK_TRUSTED, NULL); 315cd82e61cSMarcel Holtmann kfree_skb(skb_copy); 316cd82e61cSMarcel Holtmann } 317cd82e61cSMarcel Holtmann 318*38ceaa00SMarcel Holtmann void hci_send_monitor_ctrl_event(struct hci_dev *hdev, u16 event, 319*38ceaa00SMarcel Holtmann void *data, u16 data_len, ktime_t tstamp, 320*38ceaa00SMarcel Holtmann int flag, struct sock *skip_sk) 321*38ceaa00SMarcel Holtmann { 322*38ceaa00SMarcel Holtmann struct sock *sk; 323*38ceaa00SMarcel Holtmann __le16 index; 324*38ceaa00SMarcel Holtmann 325*38ceaa00SMarcel Holtmann if (hdev) 326*38ceaa00SMarcel Holtmann index = cpu_to_le16(hdev->id); 327*38ceaa00SMarcel Holtmann else 328*38ceaa00SMarcel Holtmann index = cpu_to_le16(MGMT_INDEX_NONE); 329*38ceaa00SMarcel Holtmann 330*38ceaa00SMarcel Holtmann read_lock(&hci_sk_list.lock); 331*38ceaa00SMarcel Holtmann 332*38ceaa00SMarcel Holtmann sk_for_each(sk, &hci_sk_list.head) { 333*38ceaa00SMarcel Holtmann struct hci_mon_hdr *hdr; 334*38ceaa00SMarcel Holtmann struct sk_buff *skb; 335*38ceaa00SMarcel Holtmann 336*38ceaa00SMarcel Holtmann if (hci_pi(sk)->channel != HCI_CHANNEL_CONTROL) 337*38ceaa00SMarcel Holtmann continue; 338*38ceaa00SMarcel Holtmann 339*38ceaa00SMarcel Holtmann /* Ignore socket without the flag set */ 340*38ceaa00SMarcel Holtmann if (!hci_sock_test_flag(sk, flag)) 341*38ceaa00SMarcel Holtmann continue; 342*38ceaa00SMarcel Holtmann 343*38ceaa00SMarcel Holtmann /* Skip the original socket */ 344*38ceaa00SMarcel Holtmann if (sk == skip_sk) 345*38ceaa00SMarcel Holtmann continue; 346*38ceaa00SMarcel Holtmann 347*38ceaa00SMarcel Holtmann skb = bt_skb_alloc(6 + data_len, GFP_ATOMIC); 348*38ceaa00SMarcel Holtmann if (!skb) 349*38ceaa00SMarcel Holtmann continue; 350*38ceaa00SMarcel Holtmann 351*38ceaa00SMarcel Holtmann put_unaligned_le32(hci_pi(sk)->cookie, skb_put(skb, 4)); 352*38ceaa00SMarcel Holtmann put_unaligned_le16(event, skb_put(skb, 2)); 353*38ceaa00SMarcel Holtmann 354*38ceaa00SMarcel Holtmann if (data) 355*38ceaa00SMarcel Holtmann memcpy(skb_put(skb, data_len), data, data_len); 356*38ceaa00SMarcel Holtmann 357*38ceaa00SMarcel Holtmann skb->tstamp = tstamp; 358*38ceaa00SMarcel Holtmann 359*38ceaa00SMarcel Holtmann hdr = (void *)skb_push(skb, HCI_MON_HDR_SIZE); 360*38ceaa00SMarcel Holtmann hdr->opcode = cpu_to_le16(HCI_MON_CTRL_EVENT); 361*38ceaa00SMarcel Holtmann hdr->index = index; 362*38ceaa00SMarcel Holtmann hdr->len = cpu_to_le16(skb->len - HCI_MON_HDR_SIZE); 363*38ceaa00SMarcel Holtmann 364*38ceaa00SMarcel Holtmann hci_send_to_channel(HCI_CHANNEL_MONITOR, skb, 365*38ceaa00SMarcel Holtmann HCI_SOCK_TRUSTED, NULL); 366*38ceaa00SMarcel Holtmann kfree_skb(skb); 367*38ceaa00SMarcel Holtmann } 368*38ceaa00SMarcel Holtmann 369*38ceaa00SMarcel Holtmann read_unlock(&hci_sk_list.lock); 370*38ceaa00SMarcel Holtmann } 371*38ceaa00SMarcel Holtmann 372cd82e61cSMarcel Holtmann static struct sk_buff *create_monitor_event(struct hci_dev *hdev, int event) 373cd82e61cSMarcel Holtmann { 374cd82e61cSMarcel Holtmann struct hci_mon_hdr *hdr; 375cd82e61cSMarcel Holtmann struct hci_mon_new_index *ni; 3766c566dd5SMarcel Holtmann struct hci_mon_index_info *ii; 377cd82e61cSMarcel Holtmann struct sk_buff *skb; 378cd82e61cSMarcel Holtmann __le16 opcode; 379cd82e61cSMarcel Holtmann 380cd82e61cSMarcel Holtmann switch (event) { 381cd82e61cSMarcel Holtmann case HCI_DEV_REG: 382cd82e61cSMarcel Holtmann skb = bt_skb_alloc(HCI_MON_NEW_INDEX_SIZE, GFP_ATOMIC); 383cd82e61cSMarcel Holtmann if (!skb) 384cd82e61cSMarcel Holtmann return NULL; 385cd82e61cSMarcel Holtmann 386cd82e61cSMarcel Holtmann ni = (void *)skb_put(skb, HCI_MON_NEW_INDEX_SIZE); 387cd82e61cSMarcel Holtmann ni->type = hdev->dev_type; 388cd82e61cSMarcel Holtmann ni->bus = hdev->bus; 389cd82e61cSMarcel Holtmann bacpy(&ni->bdaddr, &hdev->bdaddr); 390cd82e61cSMarcel Holtmann memcpy(ni->name, hdev->name, 8); 391cd82e61cSMarcel Holtmann 392dcf4adbfSJoe Perches opcode = cpu_to_le16(HCI_MON_NEW_INDEX); 393cd82e61cSMarcel Holtmann break; 394cd82e61cSMarcel Holtmann 395cd82e61cSMarcel Holtmann case HCI_DEV_UNREG: 396cd82e61cSMarcel Holtmann skb = bt_skb_alloc(0, GFP_ATOMIC); 397cd82e61cSMarcel Holtmann if (!skb) 398cd82e61cSMarcel Holtmann return NULL; 399cd82e61cSMarcel Holtmann 400dcf4adbfSJoe Perches opcode = cpu_to_le16(HCI_MON_DEL_INDEX); 401cd82e61cSMarcel Holtmann break; 402cd82e61cSMarcel Holtmann 403e131d74aSMarcel Holtmann case HCI_DEV_SETUP: 404e131d74aSMarcel Holtmann if (hdev->manufacturer == 0xffff) 405e131d74aSMarcel Holtmann return NULL; 406e131d74aSMarcel Holtmann 407e131d74aSMarcel Holtmann /* fall through */ 408e131d74aSMarcel Holtmann 4096c566dd5SMarcel Holtmann case HCI_DEV_UP: 4106c566dd5SMarcel Holtmann skb = bt_skb_alloc(HCI_MON_INDEX_INFO_SIZE, GFP_ATOMIC); 4116c566dd5SMarcel Holtmann if (!skb) 4126c566dd5SMarcel Holtmann return NULL; 4136c566dd5SMarcel Holtmann 4146c566dd5SMarcel Holtmann ii = (void *)skb_put(skb, HCI_MON_INDEX_INFO_SIZE); 4156c566dd5SMarcel Holtmann bacpy(&ii->bdaddr, &hdev->bdaddr); 4166c566dd5SMarcel Holtmann ii->manufacturer = cpu_to_le16(hdev->manufacturer); 4176c566dd5SMarcel Holtmann 4186c566dd5SMarcel Holtmann opcode = cpu_to_le16(HCI_MON_INDEX_INFO); 4196c566dd5SMarcel Holtmann break; 4206c566dd5SMarcel Holtmann 42122db3cbcSMarcel Holtmann case HCI_DEV_OPEN: 42222db3cbcSMarcel Holtmann skb = bt_skb_alloc(0, GFP_ATOMIC); 42322db3cbcSMarcel Holtmann if (!skb) 42422db3cbcSMarcel Holtmann return NULL; 42522db3cbcSMarcel Holtmann 42622db3cbcSMarcel Holtmann opcode = cpu_to_le16(HCI_MON_OPEN_INDEX); 42722db3cbcSMarcel Holtmann break; 42822db3cbcSMarcel Holtmann 42922db3cbcSMarcel Holtmann case HCI_DEV_CLOSE: 43022db3cbcSMarcel Holtmann skb = bt_skb_alloc(0, GFP_ATOMIC); 43122db3cbcSMarcel Holtmann if (!skb) 43222db3cbcSMarcel Holtmann return NULL; 43322db3cbcSMarcel Holtmann 43422db3cbcSMarcel Holtmann opcode = cpu_to_le16(HCI_MON_CLOSE_INDEX); 43522db3cbcSMarcel Holtmann break; 43622db3cbcSMarcel Holtmann 437cd82e61cSMarcel Holtmann default: 438cd82e61cSMarcel Holtmann return NULL; 439cd82e61cSMarcel Holtmann } 440cd82e61cSMarcel Holtmann 441cd82e61cSMarcel Holtmann __net_timestamp(skb); 442cd82e61cSMarcel Holtmann 443cd82e61cSMarcel Holtmann hdr = (void *)skb_push(skb, HCI_MON_HDR_SIZE); 444cd82e61cSMarcel Holtmann hdr->opcode = opcode; 445cd82e61cSMarcel Holtmann hdr->index = cpu_to_le16(hdev->id); 446cd82e61cSMarcel Holtmann hdr->len = cpu_to_le16(skb->len - HCI_MON_HDR_SIZE); 447cd82e61cSMarcel Holtmann 448cd82e61cSMarcel Holtmann return skb; 449cd82e61cSMarcel Holtmann } 450cd82e61cSMarcel Holtmann 451249fa169SMarcel Holtmann static struct sk_buff *create_monitor_ctrl_open(struct sock *sk) 452249fa169SMarcel Holtmann { 453249fa169SMarcel Holtmann struct hci_mon_hdr *hdr; 454249fa169SMarcel Holtmann struct sk_buff *skb; 455249fa169SMarcel Holtmann u16 format = 0x0002; 456249fa169SMarcel Holtmann u8 ver[3]; 457249fa169SMarcel Holtmann u32 flags; 458249fa169SMarcel Holtmann 459249fa169SMarcel Holtmann skb = bt_skb_alloc(14 + TASK_COMM_LEN , GFP_ATOMIC); 460249fa169SMarcel Holtmann if (!skb) 461249fa169SMarcel Holtmann return NULL; 462249fa169SMarcel Holtmann 463249fa169SMarcel Holtmann mgmt_fill_version_info(ver); 464249fa169SMarcel Holtmann flags = hci_sock_test_flag(sk, HCI_SOCK_TRUSTED) ? 0x1 : 0x0; 465249fa169SMarcel Holtmann 466249fa169SMarcel Holtmann put_unaligned_le32(hci_pi(sk)->cookie, skb_put(skb, 4)); 467249fa169SMarcel Holtmann put_unaligned_le16(format, skb_put(skb, 2)); 468249fa169SMarcel Holtmann memcpy(skb_put(skb, sizeof(ver)), ver, sizeof(ver)); 469249fa169SMarcel Holtmann put_unaligned_le32(flags, skb_put(skb, 4)); 470249fa169SMarcel Holtmann *skb_put(skb, 1) = TASK_COMM_LEN; 471249fa169SMarcel Holtmann memcpy(skb_put(skb, TASK_COMM_LEN), hci_pi(sk)->comm, TASK_COMM_LEN); 472249fa169SMarcel Holtmann 473249fa169SMarcel Holtmann __net_timestamp(skb); 474249fa169SMarcel Holtmann 475249fa169SMarcel Holtmann hdr = (void *)skb_push(skb, HCI_MON_HDR_SIZE); 476249fa169SMarcel Holtmann hdr->opcode = cpu_to_le16(HCI_MON_CTRL_OPEN); 477249fa169SMarcel Holtmann hdr->index = cpu_to_le16(HCI_DEV_NONE); 478249fa169SMarcel Holtmann hdr->len = cpu_to_le16(skb->len - HCI_MON_HDR_SIZE); 479249fa169SMarcel Holtmann 480249fa169SMarcel Holtmann return skb; 481249fa169SMarcel Holtmann } 482249fa169SMarcel Holtmann 483249fa169SMarcel Holtmann static struct sk_buff *create_monitor_ctrl_close(struct sock *sk) 484249fa169SMarcel Holtmann { 485249fa169SMarcel Holtmann struct hci_mon_hdr *hdr; 486249fa169SMarcel Holtmann struct sk_buff *skb; 487249fa169SMarcel Holtmann 488249fa169SMarcel Holtmann skb = bt_skb_alloc(4, GFP_ATOMIC); 489249fa169SMarcel Holtmann if (!skb) 490249fa169SMarcel Holtmann return NULL; 491249fa169SMarcel Holtmann 492249fa169SMarcel Holtmann put_unaligned_le32(hci_pi(sk)->cookie, skb_put(skb, 4)); 493249fa169SMarcel Holtmann 494249fa169SMarcel Holtmann __net_timestamp(skb); 495249fa169SMarcel Holtmann 496249fa169SMarcel Holtmann hdr = (void *)skb_push(skb, HCI_MON_HDR_SIZE); 497249fa169SMarcel Holtmann hdr->opcode = cpu_to_le16(HCI_MON_CTRL_CLOSE); 498249fa169SMarcel Holtmann hdr->index = cpu_to_le16(HCI_DEV_NONE); 499249fa169SMarcel Holtmann hdr->len = cpu_to_le16(skb->len - HCI_MON_HDR_SIZE); 500249fa169SMarcel Holtmann 501249fa169SMarcel Holtmann return skb; 502249fa169SMarcel Holtmann } 503249fa169SMarcel Holtmann 504*38ceaa00SMarcel Holtmann static struct sk_buff *create_monitor_ctrl_command(struct sock *sk, u16 index, 505*38ceaa00SMarcel Holtmann u16 opcode, u16 len, 506*38ceaa00SMarcel Holtmann const void *buf) 507*38ceaa00SMarcel Holtmann { 508*38ceaa00SMarcel Holtmann struct hci_mon_hdr *hdr; 509*38ceaa00SMarcel Holtmann struct sk_buff *skb; 510*38ceaa00SMarcel Holtmann 511*38ceaa00SMarcel Holtmann skb = bt_skb_alloc(6 + len, GFP_ATOMIC); 512*38ceaa00SMarcel Holtmann if (!skb) 513*38ceaa00SMarcel Holtmann return NULL; 514*38ceaa00SMarcel Holtmann 515*38ceaa00SMarcel Holtmann put_unaligned_le32(hci_pi(sk)->cookie, skb_put(skb, 4)); 516*38ceaa00SMarcel Holtmann put_unaligned_le16(opcode, skb_put(skb, 2)); 517*38ceaa00SMarcel Holtmann 518*38ceaa00SMarcel Holtmann if (buf) 519*38ceaa00SMarcel Holtmann memcpy(skb_put(skb, len), buf, len); 520*38ceaa00SMarcel Holtmann 521*38ceaa00SMarcel Holtmann __net_timestamp(skb); 522*38ceaa00SMarcel Holtmann 523*38ceaa00SMarcel Holtmann hdr = (void *)skb_push(skb, HCI_MON_HDR_SIZE); 524*38ceaa00SMarcel Holtmann hdr->opcode = cpu_to_le16(HCI_MON_CTRL_COMMAND); 525*38ceaa00SMarcel Holtmann hdr->index = cpu_to_le16(index); 526*38ceaa00SMarcel Holtmann hdr->len = cpu_to_le16(skb->len - HCI_MON_HDR_SIZE); 527*38ceaa00SMarcel Holtmann 528*38ceaa00SMarcel Holtmann return skb; 529*38ceaa00SMarcel Holtmann } 530*38ceaa00SMarcel Holtmann 531787b306cSJohannes Berg static void __printf(2, 3) 532787b306cSJohannes Berg send_monitor_note(struct sock *sk, const char *fmt, ...) 533dd31506dSMarcel Holtmann { 534787b306cSJohannes Berg size_t len; 535dd31506dSMarcel Holtmann struct hci_mon_hdr *hdr; 536dd31506dSMarcel Holtmann struct sk_buff *skb; 537787b306cSJohannes Berg va_list args; 538787b306cSJohannes Berg 539787b306cSJohannes Berg va_start(args, fmt); 540787b306cSJohannes Berg len = vsnprintf(NULL, 0, fmt, args); 541787b306cSJohannes Berg va_end(args); 542dd31506dSMarcel Holtmann 543dd31506dSMarcel Holtmann skb = bt_skb_alloc(len + 1, GFP_ATOMIC); 544dd31506dSMarcel Holtmann if (!skb) 545dd31506dSMarcel Holtmann return; 546dd31506dSMarcel Holtmann 547787b306cSJohannes Berg va_start(args, fmt); 548787b306cSJohannes Berg vsprintf(skb_put(skb, len), fmt, args); 549787b306cSJohannes Berg *skb_put(skb, 1) = 0; 550787b306cSJohannes Berg va_end(args); 551dd31506dSMarcel Holtmann 552dd31506dSMarcel Holtmann __net_timestamp(skb); 553dd31506dSMarcel Holtmann 554dd31506dSMarcel Holtmann hdr = (void *)skb_push(skb, HCI_MON_HDR_SIZE); 555dd31506dSMarcel Holtmann hdr->opcode = cpu_to_le16(HCI_MON_SYSTEM_NOTE); 556dd31506dSMarcel Holtmann hdr->index = cpu_to_le16(HCI_DEV_NONE); 557dd31506dSMarcel Holtmann hdr->len = cpu_to_le16(skb->len - HCI_MON_HDR_SIZE); 558dd31506dSMarcel Holtmann 559dd31506dSMarcel Holtmann if (sock_queue_rcv_skb(sk, skb)) 560dd31506dSMarcel Holtmann kfree_skb(skb); 561dd31506dSMarcel Holtmann } 562dd31506dSMarcel Holtmann 563cd82e61cSMarcel Holtmann static void send_monitor_replay(struct sock *sk) 564cd82e61cSMarcel Holtmann { 565cd82e61cSMarcel Holtmann struct hci_dev *hdev; 566cd82e61cSMarcel Holtmann 567cd82e61cSMarcel Holtmann read_lock(&hci_dev_list_lock); 568cd82e61cSMarcel Holtmann 569cd82e61cSMarcel Holtmann list_for_each_entry(hdev, &hci_dev_list, list) { 570cd82e61cSMarcel Holtmann struct sk_buff *skb; 571cd82e61cSMarcel Holtmann 572cd82e61cSMarcel Holtmann skb = create_monitor_event(hdev, HCI_DEV_REG); 573cd82e61cSMarcel Holtmann if (!skb) 574cd82e61cSMarcel Holtmann continue; 575cd82e61cSMarcel Holtmann 576cd82e61cSMarcel Holtmann if (sock_queue_rcv_skb(sk, skb)) 577cd82e61cSMarcel Holtmann kfree_skb(skb); 57822db3cbcSMarcel Holtmann 57922db3cbcSMarcel Holtmann if (!test_bit(HCI_RUNNING, &hdev->flags)) 58022db3cbcSMarcel Holtmann continue; 58122db3cbcSMarcel Holtmann 58222db3cbcSMarcel Holtmann skb = create_monitor_event(hdev, HCI_DEV_OPEN); 58322db3cbcSMarcel Holtmann if (!skb) 58422db3cbcSMarcel Holtmann continue; 58522db3cbcSMarcel Holtmann 58622db3cbcSMarcel Holtmann if (sock_queue_rcv_skb(sk, skb)) 58722db3cbcSMarcel Holtmann kfree_skb(skb); 5886c566dd5SMarcel Holtmann 589e131d74aSMarcel Holtmann if (test_bit(HCI_UP, &hdev->flags)) 5906c566dd5SMarcel Holtmann skb = create_monitor_event(hdev, HCI_DEV_UP); 591e131d74aSMarcel Holtmann else if (hci_dev_test_flag(hdev, HCI_SETUP)) 592e131d74aSMarcel Holtmann skb = create_monitor_event(hdev, HCI_DEV_SETUP); 593e131d74aSMarcel Holtmann else 594e131d74aSMarcel Holtmann skb = NULL; 5956c566dd5SMarcel Holtmann 596e131d74aSMarcel Holtmann if (skb) { 5976c566dd5SMarcel Holtmann if (sock_queue_rcv_skb(sk, skb)) 5986c566dd5SMarcel Holtmann kfree_skb(skb); 599cd82e61cSMarcel Holtmann } 600e131d74aSMarcel Holtmann } 601cd82e61cSMarcel Holtmann 602cd82e61cSMarcel Holtmann read_unlock(&hci_dev_list_lock); 603cd82e61cSMarcel Holtmann } 604cd82e61cSMarcel Holtmann 605249fa169SMarcel Holtmann static void send_monitor_control_replay(struct sock *mon_sk) 606249fa169SMarcel Holtmann { 607249fa169SMarcel Holtmann struct sock *sk; 608249fa169SMarcel Holtmann 609249fa169SMarcel Holtmann read_lock(&hci_sk_list.lock); 610249fa169SMarcel Holtmann 611249fa169SMarcel Holtmann sk_for_each(sk, &hci_sk_list.head) { 612249fa169SMarcel Holtmann struct sk_buff *skb; 613249fa169SMarcel Holtmann 614249fa169SMarcel Holtmann if (hci_pi(sk)->channel != HCI_CHANNEL_CONTROL) 615249fa169SMarcel Holtmann continue; 616249fa169SMarcel Holtmann 617249fa169SMarcel Holtmann skb = create_monitor_ctrl_open(sk); 618249fa169SMarcel Holtmann if (!skb) 619249fa169SMarcel Holtmann continue; 620249fa169SMarcel Holtmann 621249fa169SMarcel Holtmann if (sock_queue_rcv_skb(mon_sk, skb)) 622249fa169SMarcel Holtmann kfree_skb(skb); 623249fa169SMarcel Holtmann } 624249fa169SMarcel Holtmann 625249fa169SMarcel Holtmann read_unlock(&hci_sk_list.lock); 626249fa169SMarcel Holtmann } 627249fa169SMarcel Holtmann 628040030efSMarcel Holtmann /* Generate internal stack event */ 629040030efSMarcel Holtmann static void hci_si_event(struct hci_dev *hdev, int type, int dlen, void *data) 630040030efSMarcel Holtmann { 631040030efSMarcel Holtmann struct hci_event_hdr *hdr; 632040030efSMarcel Holtmann struct hci_ev_stack_internal *ev; 633040030efSMarcel Holtmann struct sk_buff *skb; 634040030efSMarcel Holtmann 635040030efSMarcel Holtmann skb = bt_skb_alloc(HCI_EVENT_HDR_SIZE + sizeof(*ev) + dlen, GFP_ATOMIC); 636040030efSMarcel Holtmann if (!skb) 637040030efSMarcel Holtmann return; 638040030efSMarcel Holtmann 639040030efSMarcel Holtmann hdr = (void *)skb_put(skb, HCI_EVENT_HDR_SIZE); 640040030efSMarcel Holtmann hdr->evt = HCI_EV_STACK_INTERNAL; 641040030efSMarcel Holtmann hdr->plen = sizeof(*ev) + dlen; 642040030efSMarcel Holtmann 643040030efSMarcel Holtmann ev = (void *)skb_put(skb, sizeof(*ev) + dlen); 644040030efSMarcel Holtmann ev->type = type; 645040030efSMarcel Holtmann memcpy(ev->data, data, dlen); 646040030efSMarcel Holtmann 647040030efSMarcel Holtmann bt_cb(skb)->incoming = 1; 648040030efSMarcel Holtmann __net_timestamp(skb); 649040030efSMarcel Holtmann 650d79f34e3SMarcel Holtmann hci_skb_pkt_type(skb) = HCI_EVENT_PKT; 651040030efSMarcel Holtmann hci_send_to_sock(hdev, skb); 652040030efSMarcel Holtmann kfree_skb(skb); 653040030efSMarcel Holtmann } 654040030efSMarcel Holtmann 655040030efSMarcel Holtmann void hci_sock_dev_event(struct hci_dev *hdev, int event) 656040030efSMarcel Holtmann { 657040030efSMarcel Holtmann BT_DBG("hdev %s event %d", hdev->name, event); 658040030efSMarcel Holtmann 659cd82e61cSMarcel Holtmann if (atomic_read(&monitor_promisc)) { 660cd82e61cSMarcel Holtmann struct sk_buff *skb; 661cd82e61cSMarcel Holtmann 662ed1b28a4SMarcel Holtmann /* Send event to monitor */ 663cd82e61cSMarcel Holtmann skb = create_monitor_event(hdev, event); 664cd82e61cSMarcel Holtmann if (skb) { 665c08b1a1dSMarcel Holtmann hci_send_to_channel(HCI_CHANNEL_MONITOR, skb, 666c08b1a1dSMarcel Holtmann HCI_SOCK_TRUSTED, NULL); 667cd82e61cSMarcel Holtmann kfree_skb(skb); 668cd82e61cSMarcel Holtmann } 669cd82e61cSMarcel Holtmann } 670cd82e61cSMarcel Holtmann 671ed1b28a4SMarcel Holtmann if (event <= HCI_DEV_DOWN) { 672ed1b28a4SMarcel Holtmann struct hci_ev_si_device ev; 673ed1b28a4SMarcel Holtmann 674040030efSMarcel Holtmann /* Send event to sockets */ 675040030efSMarcel Holtmann ev.event = event; 676040030efSMarcel Holtmann ev.dev_id = hdev->id; 677040030efSMarcel Holtmann hci_si_event(NULL, HCI_EV_SI_DEVICE, sizeof(ev), &ev); 678ed1b28a4SMarcel Holtmann } 679040030efSMarcel Holtmann 680040030efSMarcel Holtmann if (event == HCI_DEV_UNREG) { 681040030efSMarcel Holtmann struct sock *sk; 682040030efSMarcel Holtmann 683040030efSMarcel Holtmann /* Detach sockets from device */ 684040030efSMarcel Holtmann read_lock(&hci_sk_list.lock); 685b67bfe0dSSasha Levin sk_for_each(sk, &hci_sk_list.head) { 686040030efSMarcel Holtmann bh_lock_sock_nested(sk); 687040030efSMarcel Holtmann if (hci_pi(sk)->hdev == hdev) { 688040030efSMarcel Holtmann hci_pi(sk)->hdev = NULL; 689040030efSMarcel Holtmann sk->sk_err = EPIPE; 690040030efSMarcel Holtmann sk->sk_state = BT_OPEN; 691040030efSMarcel Holtmann sk->sk_state_change(sk); 692040030efSMarcel Holtmann 693040030efSMarcel Holtmann hci_dev_put(hdev); 694040030efSMarcel Holtmann } 695040030efSMarcel Holtmann bh_unlock_sock(sk); 696040030efSMarcel Holtmann } 697040030efSMarcel Holtmann read_unlock(&hci_sk_list.lock); 698040030efSMarcel Holtmann } 699040030efSMarcel Holtmann } 700040030efSMarcel Holtmann 701801c1e8dSJohan Hedberg static struct hci_mgmt_chan *__hci_mgmt_chan_find(unsigned short channel) 702801c1e8dSJohan Hedberg { 703801c1e8dSJohan Hedberg struct hci_mgmt_chan *c; 704801c1e8dSJohan Hedberg 705801c1e8dSJohan Hedberg list_for_each_entry(c, &mgmt_chan_list, list) { 706801c1e8dSJohan Hedberg if (c->channel == channel) 707801c1e8dSJohan Hedberg return c; 708801c1e8dSJohan Hedberg } 709801c1e8dSJohan Hedberg 710801c1e8dSJohan Hedberg return NULL; 711801c1e8dSJohan Hedberg } 712801c1e8dSJohan Hedberg 713801c1e8dSJohan Hedberg static struct hci_mgmt_chan *hci_mgmt_chan_find(unsigned short channel) 714801c1e8dSJohan Hedberg { 715801c1e8dSJohan Hedberg struct hci_mgmt_chan *c; 716801c1e8dSJohan Hedberg 717801c1e8dSJohan Hedberg mutex_lock(&mgmt_chan_list_lock); 718801c1e8dSJohan Hedberg c = __hci_mgmt_chan_find(channel); 719801c1e8dSJohan Hedberg mutex_unlock(&mgmt_chan_list_lock); 720801c1e8dSJohan Hedberg 721801c1e8dSJohan Hedberg return c; 722801c1e8dSJohan Hedberg } 723801c1e8dSJohan Hedberg 724801c1e8dSJohan Hedberg int hci_mgmt_chan_register(struct hci_mgmt_chan *c) 725801c1e8dSJohan Hedberg { 726801c1e8dSJohan Hedberg if (c->channel < HCI_CHANNEL_CONTROL) 727801c1e8dSJohan Hedberg return -EINVAL; 728801c1e8dSJohan Hedberg 729801c1e8dSJohan Hedberg mutex_lock(&mgmt_chan_list_lock); 730801c1e8dSJohan Hedberg if (__hci_mgmt_chan_find(c->channel)) { 731801c1e8dSJohan Hedberg mutex_unlock(&mgmt_chan_list_lock); 732801c1e8dSJohan Hedberg return -EALREADY; 733801c1e8dSJohan Hedberg } 734801c1e8dSJohan Hedberg 735801c1e8dSJohan Hedberg list_add_tail(&c->list, &mgmt_chan_list); 736801c1e8dSJohan Hedberg 737801c1e8dSJohan Hedberg mutex_unlock(&mgmt_chan_list_lock); 738801c1e8dSJohan Hedberg 739801c1e8dSJohan Hedberg return 0; 740801c1e8dSJohan Hedberg } 741801c1e8dSJohan Hedberg EXPORT_SYMBOL(hci_mgmt_chan_register); 742801c1e8dSJohan Hedberg 743801c1e8dSJohan Hedberg void hci_mgmt_chan_unregister(struct hci_mgmt_chan *c) 744801c1e8dSJohan Hedberg { 745801c1e8dSJohan Hedberg mutex_lock(&mgmt_chan_list_lock); 746801c1e8dSJohan Hedberg list_del(&c->list); 747801c1e8dSJohan Hedberg mutex_unlock(&mgmt_chan_list_lock); 748801c1e8dSJohan Hedberg } 749801c1e8dSJohan Hedberg EXPORT_SYMBOL(hci_mgmt_chan_unregister); 750801c1e8dSJohan Hedberg 7511da177e4SLinus Torvalds static int hci_sock_release(struct socket *sock) 7521da177e4SLinus Torvalds { 7531da177e4SLinus Torvalds struct sock *sk = sock->sk; 7547b005bd3SMarcel Holtmann struct hci_dev *hdev; 755249fa169SMarcel Holtmann struct sk_buff *skb; 75670ecce91SMarcel Holtmann int id; 7571da177e4SLinus Torvalds 7581da177e4SLinus Torvalds BT_DBG("sock %p sk %p", sock, sk); 7591da177e4SLinus Torvalds 7601da177e4SLinus Torvalds if (!sk) 7611da177e4SLinus Torvalds return 0; 7621da177e4SLinus Torvalds 7637b005bd3SMarcel Holtmann hdev = hci_pi(sk)->hdev; 7647b005bd3SMarcel Holtmann 76570ecce91SMarcel Holtmann switch (hci_pi(sk)->channel) { 76670ecce91SMarcel Holtmann case HCI_CHANNEL_MONITOR: 767cd82e61cSMarcel Holtmann atomic_dec(&monitor_promisc); 76870ecce91SMarcel Holtmann break; 76970ecce91SMarcel Holtmann case HCI_CHANNEL_CONTROL: 77070ecce91SMarcel Holtmann id = hci_pi(sk)->cookie; 77170ecce91SMarcel Holtmann 772249fa169SMarcel Holtmann /* Send event to monitor */ 773249fa169SMarcel Holtmann skb = create_monitor_ctrl_close(sk); 774249fa169SMarcel Holtmann if (skb) { 775249fa169SMarcel Holtmann hci_send_to_channel(HCI_CHANNEL_MONITOR, skb, 776249fa169SMarcel Holtmann HCI_SOCK_TRUSTED, NULL); 777249fa169SMarcel Holtmann kfree_skb(skb); 778249fa169SMarcel Holtmann } 779249fa169SMarcel Holtmann 78070ecce91SMarcel Holtmann hci_pi(sk)->cookie = 0xffffffff; 78170ecce91SMarcel Holtmann ida_simple_remove(&sock_cookie_ida, id); 78270ecce91SMarcel Holtmann break; 78370ecce91SMarcel Holtmann } 784cd82e61cSMarcel Holtmann 7851da177e4SLinus Torvalds bt_sock_unlink(&hci_sk_list, sk); 7861da177e4SLinus Torvalds 7871da177e4SLinus Torvalds if (hdev) { 78823500189SMarcel Holtmann if (hci_pi(sk)->channel == HCI_CHANNEL_USER) { 7896b3cc1dbSSimon Fels /* When releasing an user channel exclusive access, 7906b3cc1dbSSimon Fels * call hci_dev_do_close directly instead of calling 7916b3cc1dbSSimon Fels * hci_dev_close to ensure the exclusive access will 7926b3cc1dbSSimon Fels * be released and the controller brought back down. 7936b3cc1dbSSimon Fels * 7946b3cc1dbSSimon Fels * The checking of HCI_AUTO_OFF is not needed in this 7956b3cc1dbSSimon Fels * case since it will have been cleared already when 7966b3cc1dbSSimon Fels * opening the user channel. 7976b3cc1dbSSimon Fels */ 7986b3cc1dbSSimon Fels hci_dev_do_close(hdev); 7999380f9eaSLoic Poulain hci_dev_clear_flag(hdev, HCI_USER_CHANNEL); 8009380f9eaSLoic Poulain mgmt_index_added(hdev); 80123500189SMarcel Holtmann } 80223500189SMarcel Holtmann 8031da177e4SLinus Torvalds atomic_dec(&hdev->promisc); 8041da177e4SLinus Torvalds hci_dev_put(hdev); 8051da177e4SLinus Torvalds } 8061da177e4SLinus Torvalds 8071da177e4SLinus Torvalds sock_orphan(sk); 8081da177e4SLinus Torvalds 8091da177e4SLinus Torvalds skb_queue_purge(&sk->sk_receive_queue); 8101da177e4SLinus Torvalds skb_queue_purge(&sk->sk_write_queue); 8111da177e4SLinus Torvalds 8121da177e4SLinus Torvalds sock_put(sk); 8131da177e4SLinus Torvalds return 0; 8141da177e4SLinus Torvalds } 8151da177e4SLinus Torvalds 816b2a66aadSAntti Julku static int hci_sock_blacklist_add(struct hci_dev *hdev, void __user *arg) 817f0358568SJohan Hedberg { 818f0358568SJohan Hedberg bdaddr_t bdaddr; 8195e762444SAntti Julku int err; 820f0358568SJohan Hedberg 821f0358568SJohan Hedberg if (copy_from_user(&bdaddr, arg, sizeof(bdaddr))) 822f0358568SJohan Hedberg return -EFAULT; 823f0358568SJohan Hedberg 82409fd0de5SGustavo F. Padovan hci_dev_lock(hdev); 8255e762444SAntti Julku 826dcc36c16SJohan Hedberg err = hci_bdaddr_list_add(&hdev->blacklist, &bdaddr, BDADDR_BREDR); 8275e762444SAntti Julku 82809fd0de5SGustavo F. Padovan hci_dev_unlock(hdev); 8295e762444SAntti Julku 8305e762444SAntti Julku return err; 831f0358568SJohan Hedberg } 832f0358568SJohan Hedberg 833b2a66aadSAntti Julku static int hci_sock_blacklist_del(struct hci_dev *hdev, void __user *arg) 834f0358568SJohan Hedberg { 835f0358568SJohan Hedberg bdaddr_t bdaddr; 8365e762444SAntti Julku int err; 837f0358568SJohan Hedberg 838f0358568SJohan Hedberg if (copy_from_user(&bdaddr, arg, sizeof(bdaddr))) 839f0358568SJohan Hedberg return -EFAULT; 840f0358568SJohan Hedberg 84109fd0de5SGustavo F. Padovan hci_dev_lock(hdev); 8425e762444SAntti Julku 843dcc36c16SJohan Hedberg err = hci_bdaddr_list_del(&hdev->blacklist, &bdaddr, BDADDR_BREDR); 8445e762444SAntti Julku 84509fd0de5SGustavo F. Padovan hci_dev_unlock(hdev); 8465e762444SAntti Julku 8475e762444SAntti Julku return err; 848f0358568SJohan Hedberg } 849f0358568SJohan Hedberg 8501da177e4SLinus Torvalds /* Ioctls that require bound socket */ 8516039aa73SGustavo Padovan static int hci_sock_bound_ioctl(struct sock *sk, unsigned int cmd, 8526039aa73SGustavo Padovan unsigned long arg) 8531da177e4SLinus Torvalds { 8541da177e4SLinus Torvalds struct hci_dev *hdev = hci_pi(sk)->hdev; 8551da177e4SLinus Torvalds 8561da177e4SLinus Torvalds if (!hdev) 8571da177e4SLinus Torvalds return -EBADFD; 8581da177e4SLinus Torvalds 859d7a5a11dSMarcel Holtmann if (hci_dev_test_flag(hdev, HCI_USER_CHANNEL)) 8600736cfa8SMarcel Holtmann return -EBUSY; 8610736cfa8SMarcel Holtmann 862d7a5a11dSMarcel Holtmann if (hci_dev_test_flag(hdev, HCI_UNCONFIGURED)) 863fee746b0SMarcel Holtmann return -EOPNOTSUPP; 864fee746b0SMarcel Holtmann 865ca8bee5dSMarcel Holtmann if (hdev->dev_type != HCI_PRIMARY) 8665b69bef5SMarcel Holtmann return -EOPNOTSUPP; 8675b69bef5SMarcel Holtmann 8681da177e4SLinus Torvalds switch (cmd) { 8691da177e4SLinus Torvalds case HCISETRAW: 8701da177e4SLinus Torvalds if (!capable(CAP_NET_ADMIN)) 871bf5b30b8SZhao Hongjiang return -EPERM; 872db596681SMarcel Holtmann return -EOPNOTSUPP; 8731da177e4SLinus Torvalds 8741da177e4SLinus Torvalds case HCIGETCONNINFO: 8751da177e4SLinus Torvalds return hci_get_conn_info(hdev, (void __user *)arg); 8761da177e4SLinus Torvalds 87740be492fSMarcel Holtmann case HCIGETAUTHINFO: 87840be492fSMarcel Holtmann return hci_get_auth_info(hdev, (void __user *)arg); 87940be492fSMarcel Holtmann 880f0358568SJohan Hedberg case HCIBLOCKADDR: 881f0358568SJohan Hedberg if (!capable(CAP_NET_ADMIN)) 882bf5b30b8SZhao Hongjiang return -EPERM; 883b2a66aadSAntti Julku return hci_sock_blacklist_add(hdev, (void __user *)arg); 884f0358568SJohan Hedberg 885f0358568SJohan Hedberg case HCIUNBLOCKADDR: 886f0358568SJohan Hedberg if (!capable(CAP_NET_ADMIN)) 887bf5b30b8SZhao Hongjiang return -EPERM; 888b2a66aadSAntti Julku return hci_sock_blacklist_del(hdev, (void __user *)arg); 8890736cfa8SMarcel Holtmann } 890f0358568SJohan Hedberg 891324d36edSMarcel Holtmann return -ENOIOCTLCMD; 8921da177e4SLinus Torvalds } 8931da177e4SLinus Torvalds 8948fc9ced3SGustavo Padovan static int hci_sock_ioctl(struct socket *sock, unsigned int cmd, 8958fc9ced3SGustavo Padovan unsigned long arg) 8961da177e4SLinus Torvalds { 8971da177e4SLinus Torvalds void __user *argp = (void __user *)arg; 8980736cfa8SMarcel Holtmann struct sock *sk = sock->sk; 8991da177e4SLinus Torvalds int err; 9001da177e4SLinus Torvalds 9011da177e4SLinus Torvalds BT_DBG("cmd %x arg %lx", cmd, arg); 9021da177e4SLinus Torvalds 903c1c4f956SMarcel Holtmann lock_sock(sk); 904c1c4f956SMarcel Holtmann 905c1c4f956SMarcel Holtmann if (hci_pi(sk)->channel != HCI_CHANNEL_RAW) { 906c1c4f956SMarcel Holtmann err = -EBADFD; 907c1c4f956SMarcel Holtmann goto done; 908c1c4f956SMarcel Holtmann } 909c1c4f956SMarcel Holtmann 910c1c4f956SMarcel Holtmann release_sock(sk); 911c1c4f956SMarcel Holtmann 9121da177e4SLinus Torvalds switch (cmd) { 9131da177e4SLinus Torvalds case HCIGETDEVLIST: 9141da177e4SLinus Torvalds return hci_get_dev_list(argp); 9151da177e4SLinus Torvalds 9161da177e4SLinus Torvalds case HCIGETDEVINFO: 9171da177e4SLinus Torvalds return hci_get_dev_info(argp); 9181da177e4SLinus Torvalds 9191da177e4SLinus Torvalds case HCIGETCONNLIST: 9201da177e4SLinus Torvalds return hci_get_conn_list(argp); 9211da177e4SLinus Torvalds 9221da177e4SLinus Torvalds case HCIDEVUP: 9231da177e4SLinus Torvalds if (!capable(CAP_NET_ADMIN)) 924bf5b30b8SZhao Hongjiang return -EPERM; 9251da177e4SLinus Torvalds return hci_dev_open(arg); 9261da177e4SLinus Torvalds 9271da177e4SLinus Torvalds case HCIDEVDOWN: 9281da177e4SLinus Torvalds if (!capable(CAP_NET_ADMIN)) 929bf5b30b8SZhao Hongjiang return -EPERM; 9301da177e4SLinus Torvalds return hci_dev_close(arg); 9311da177e4SLinus Torvalds 9321da177e4SLinus Torvalds case HCIDEVRESET: 9331da177e4SLinus Torvalds if (!capable(CAP_NET_ADMIN)) 934bf5b30b8SZhao Hongjiang return -EPERM; 9351da177e4SLinus Torvalds return hci_dev_reset(arg); 9361da177e4SLinus Torvalds 9371da177e4SLinus Torvalds case HCIDEVRESTAT: 9381da177e4SLinus Torvalds if (!capable(CAP_NET_ADMIN)) 939bf5b30b8SZhao Hongjiang return -EPERM; 9401da177e4SLinus Torvalds return hci_dev_reset_stat(arg); 9411da177e4SLinus Torvalds 9421da177e4SLinus Torvalds case HCISETSCAN: 9431da177e4SLinus Torvalds case HCISETAUTH: 9441da177e4SLinus Torvalds case HCISETENCRYPT: 9451da177e4SLinus Torvalds case HCISETPTYPE: 9461da177e4SLinus Torvalds case HCISETLINKPOL: 9471da177e4SLinus Torvalds case HCISETLINKMODE: 9481da177e4SLinus Torvalds case HCISETACLMTU: 9491da177e4SLinus Torvalds case HCISETSCOMTU: 9501da177e4SLinus Torvalds if (!capable(CAP_NET_ADMIN)) 951bf5b30b8SZhao Hongjiang return -EPERM; 9521da177e4SLinus Torvalds return hci_dev_cmd(cmd, argp); 9531da177e4SLinus Torvalds 9541da177e4SLinus Torvalds case HCIINQUIRY: 9551da177e4SLinus Torvalds return hci_inquiry(argp); 956c1c4f956SMarcel Holtmann } 9571da177e4SLinus Torvalds 9581da177e4SLinus Torvalds lock_sock(sk); 959c1c4f956SMarcel Holtmann 9601da177e4SLinus Torvalds err = hci_sock_bound_ioctl(sk, cmd, arg); 961c1c4f956SMarcel Holtmann 962c1c4f956SMarcel Holtmann done: 9631da177e4SLinus Torvalds release_sock(sk); 9641da177e4SLinus Torvalds return err; 9651da177e4SLinus Torvalds } 9661da177e4SLinus Torvalds 9678fc9ced3SGustavo Padovan static int hci_sock_bind(struct socket *sock, struct sockaddr *addr, 9688fc9ced3SGustavo Padovan int addr_len) 9691da177e4SLinus Torvalds { 9700381101fSJohan Hedberg struct sockaddr_hci haddr; 9711da177e4SLinus Torvalds struct sock *sk = sock->sk; 9721da177e4SLinus Torvalds struct hci_dev *hdev = NULL; 9730381101fSJohan Hedberg int len, err = 0; 9741da177e4SLinus Torvalds 9751da177e4SLinus Torvalds BT_DBG("sock %p sk %p", sock, sk); 9761da177e4SLinus Torvalds 9770381101fSJohan Hedberg if (!addr) 9780381101fSJohan Hedberg return -EINVAL; 9790381101fSJohan Hedberg 9800381101fSJohan Hedberg memset(&haddr, 0, sizeof(haddr)); 9810381101fSJohan Hedberg len = min_t(unsigned int, sizeof(haddr), addr_len); 9820381101fSJohan Hedberg memcpy(&haddr, addr, len); 9830381101fSJohan Hedberg 9840381101fSJohan Hedberg if (haddr.hci_family != AF_BLUETOOTH) 9850381101fSJohan Hedberg return -EINVAL; 9860381101fSJohan Hedberg 9871da177e4SLinus Torvalds lock_sock(sk); 9881da177e4SLinus Torvalds 9897cc2ade2SMarcel Holtmann if (sk->sk_state == BT_BOUND) { 9907cc2ade2SMarcel Holtmann err = -EALREADY; 9917cc2ade2SMarcel Holtmann goto done; 9927cc2ade2SMarcel Holtmann } 9937cc2ade2SMarcel Holtmann 9947cc2ade2SMarcel Holtmann switch (haddr.hci_channel) { 9957cc2ade2SMarcel Holtmann case HCI_CHANNEL_RAW: 9967cc2ade2SMarcel Holtmann if (hci_pi(sk)->hdev) { 9971da177e4SLinus Torvalds err = -EALREADY; 9981da177e4SLinus Torvalds goto done; 9991da177e4SLinus Torvalds } 10001da177e4SLinus Torvalds 10010381101fSJohan Hedberg if (haddr.hci_dev != HCI_DEV_NONE) { 10020381101fSJohan Hedberg hdev = hci_dev_get(haddr.hci_dev); 100370f23020SAndrei Emeltchenko if (!hdev) { 10041da177e4SLinus Torvalds err = -ENODEV; 10051da177e4SLinus Torvalds goto done; 10061da177e4SLinus Torvalds } 10071da177e4SLinus Torvalds 10081da177e4SLinus Torvalds atomic_inc(&hdev->promisc); 10091da177e4SLinus Torvalds } 10101da177e4SLinus Torvalds 10111da177e4SLinus Torvalds hci_pi(sk)->hdev = hdev; 10127cc2ade2SMarcel Holtmann break; 10137cc2ade2SMarcel Holtmann 101423500189SMarcel Holtmann case HCI_CHANNEL_USER: 101523500189SMarcel Holtmann if (hci_pi(sk)->hdev) { 101623500189SMarcel Holtmann err = -EALREADY; 101723500189SMarcel Holtmann goto done; 101823500189SMarcel Holtmann } 101923500189SMarcel Holtmann 102023500189SMarcel Holtmann if (haddr.hci_dev == HCI_DEV_NONE) { 102123500189SMarcel Holtmann err = -EINVAL; 102223500189SMarcel Holtmann goto done; 102323500189SMarcel Holtmann } 102423500189SMarcel Holtmann 102510a8b86fSMarcel Holtmann if (!capable(CAP_NET_ADMIN)) { 102623500189SMarcel Holtmann err = -EPERM; 102723500189SMarcel Holtmann goto done; 102823500189SMarcel Holtmann } 102923500189SMarcel Holtmann 103023500189SMarcel Holtmann hdev = hci_dev_get(haddr.hci_dev); 103123500189SMarcel Holtmann if (!hdev) { 103223500189SMarcel Holtmann err = -ENODEV; 103323500189SMarcel Holtmann goto done; 103423500189SMarcel Holtmann } 103523500189SMarcel Holtmann 1036781f899fSMarcel Holtmann if (test_bit(HCI_INIT, &hdev->flags) || 1037d7a5a11dSMarcel Holtmann hci_dev_test_flag(hdev, HCI_SETUP) || 1038781f899fSMarcel Holtmann hci_dev_test_flag(hdev, HCI_CONFIG) || 1039781f899fSMarcel Holtmann (!hci_dev_test_flag(hdev, HCI_AUTO_OFF) && 1040781f899fSMarcel Holtmann test_bit(HCI_UP, &hdev->flags))) { 104123500189SMarcel Holtmann err = -EBUSY; 104223500189SMarcel Holtmann hci_dev_put(hdev); 104323500189SMarcel Holtmann goto done; 104423500189SMarcel Holtmann } 104523500189SMarcel Holtmann 1046238be788SMarcel Holtmann if (hci_dev_test_and_set_flag(hdev, HCI_USER_CHANNEL)) { 104723500189SMarcel Holtmann err = -EUSERS; 104823500189SMarcel Holtmann hci_dev_put(hdev); 104923500189SMarcel Holtmann goto done; 105023500189SMarcel Holtmann } 105123500189SMarcel Holtmann 105223500189SMarcel Holtmann mgmt_index_removed(hdev); 105323500189SMarcel Holtmann 105423500189SMarcel Holtmann err = hci_dev_open(hdev->id); 105523500189SMarcel Holtmann if (err) { 1056781f899fSMarcel Holtmann if (err == -EALREADY) { 1057781f899fSMarcel Holtmann /* In case the transport is already up and 1058781f899fSMarcel Holtmann * running, clear the error here. 1059781f899fSMarcel Holtmann * 1060781f899fSMarcel Holtmann * This can happen when opening an user 1061781f899fSMarcel Holtmann * channel and HCI_AUTO_OFF grace period 1062781f899fSMarcel Holtmann * is still active. 1063781f899fSMarcel Holtmann */ 1064781f899fSMarcel Holtmann err = 0; 1065781f899fSMarcel Holtmann } else { 1066a358dc11SMarcel Holtmann hci_dev_clear_flag(hdev, HCI_USER_CHANNEL); 1067c6521401SMarcel Holtmann mgmt_index_added(hdev); 106823500189SMarcel Holtmann hci_dev_put(hdev); 106923500189SMarcel Holtmann goto done; 107023500189SMarcel Holtmann } 1071781f899fSMarcel Holtmann } 107223500189SMarcel Holtmann 107323500189SMarcel Holtmann atomic_inc(&hdev->promisc); 107423500189SMarcel Holtmann 107523500189SMarcel Holtmann hci_pi(sk)->hdev = hdev; 107623500189SMarcel Holtmann break; 107723500189SMarcel Holtmann 1078cd82e61cSMarcel Holtmann case HCI_CHANNEL_MONITOR: 1079cd82e61cSMarcel Holtmann if (haddr.hci_dev != HCI_DEV_NONE) { 1080cd82e61cSMarcel Holtmann err = -EINVAL; 1081cd82e61cSMarcel Holtmann goto done; 1082cd82e61cSMarcel Holtmann } 1083cd82e61cSMarcel Holtmann 1084cd82e61cSMarcel Holtmann if (!capable(CAP_NET_RAW)) { 1085cd82e61cSMarcel Holtmann err = -EPERM; 1086cd82e61cSMarcel Holtmann goto done; 1087cd82e61cSMarcel Holtmann } 1088cd82e61cSMarcel Holtmann 108950ebc055SMarcel Holtmann /* The monitor interface is restricted to CAP_NET_RAW 109050ebc055SMarcel Holtmann * capabilities and with that implicitly trusted. 109150ebc055SMarcel Holtmann */ 109250ebc055SMarcel Holtmann hci_sock_set_flag(sk, HCI_SOCK_TRUSTED); 109350ebc055SMarcel Holtmann 1094787b306cSJohannes Berg send_monitor_note(sk, "Linux version %s (%s)", 1095787b306cSJohannes Berg init_utsname()->release, 1096787b306cSJohannes Berg init_utsname()->machine); 1097787b306cSJohannes Berg send_monitor_note(sk, "Bluetooth subsystem version %s", 1098dd31506dSMarcel Holtmann BT_SUBSYS_VERSION); 1099cd82e61cSMarcel Holtmann send_monitor_replay(sk); 1100249fa169SMarcel Holtmann send_monitor_control_replay(sk); 1101cd82e61cSMarcel Holtmann 1102cd82e61cSMarcel Holtmann atomic_inc(&monitor_promisc); 1103cd82e61cSMarcel Holtmann break; 1104cd82e61cSMarcel Holtmann 1105ac714949SMarcel Holtmann case HCI_CHANNEL_LOGGING: 1106ac714949SMarcel Holtmann if (haddr.hci_dev != HCI_DEV_NONE) { 1107ac714949SMarcel Holtmann err = -EINVAL; 1108ac714949SMarcel Holtmann goto done; 1109ac714949SMarcel Holtmann } 1110ac714949SMarcel Holtmann 1111ac714949SMarcel Holtmann if (!capable(CAP_NET_ADMIN)) { 1112ac714949SMarcel Holtmann err = -EPERM; 1113ac714949SMarcel Holtmann goto done; 1114ac714949SMarcel Holtmann } 1115ac714949SMarcel Holtmann break; 1116ac714949SMarcel Holtmann 11177cc2ade2SMarcel Holtmann default: 1118801c1e8dSJohan Hedberg if (!hci_mgmt_chan_find(haddr.hci_channel)) { 11197cc2ade2SMarcel Holtmann err = -EINVAL; 11207cc2ade2SMarcel Holtmann goto done; 11217cc2ade2SMarcel Holtmann } 11227cc2ade2SMarcel Holtmann 1123801c1e8dSJohan Hedberg if (haddr.hci_dev != HCI_DEV_NONE) { 1124801c1e8dSJohan Hedberg err = -EINVAL; 1125801c1e8dSJohan Hedberg goto done; 1126801c1e8dSJohan Hedberg } 1127801c1e8dSJohan Hedberg 11281195fbb8SMarcel Holtmann /* Users with CAP_NET_ADMIN capabilities are allowed 11291195fbb8SMarcel Holtmann * access to all management commands and events. For 11301195fbb8SMarcel Holtmann * untrusted users the interface is restricted and 11311195fbb8SMarcel Holtmann * also only untrusted events are sent. 113250ebc055SMarcel Holtmann */ 11331195fbb8SMarcel Holtmann if (capable(CAP_NET_ADMIN)) 113450ebc055SMarcel Holtmann hci_sock_set_flag(sk, HCI_SOCK_TRUSTED); 113550ebc055SMarcel Holtmann 1136f9207338SMarcel Holtmann /* At the moment the index and unconfigured index events 1137f9207338SMarcel Holtmann * are enabled unconditionally. Setting them on each 1138f9207338SMarcel Holtmann * socket when binding keeps this functionality. They 1139f9207338SMarcel Holtmann * however might be cleared later and then sending of these 1140f9207338SMarcel Holtmann * events will be disabled, but that is then intentional. 1141f6b7712eSMarcel Holtmann * 1142f6b7712eSMarcel Holtmann * This also enables generic events that are safe to be 1143f6b7712eSMarcel Holtmann * received by untrusted users. Example for such events 1144f6b7712eSMarcel Holtmann * are changes to settings, class of device, name etc. 1145f9207338SMarcel Holtmann */ 1146f9207338SMarcel Holtmann if (haddr.hci_channel == HCI_CHANNEL_CONTROL) { 1147249fa169SMarcel Holtmann struct sk_buff *skb; 114870ecce91SMarcel Holtmann int id; 114970ecce91SMarcel Holtmann 115070ecce91SMarcel Holtmann id = ida_simple_get(&sock_cookie_ida, 1, 0, GFP_KERNEL); 115170ecce91SMarcel Holtmann if (id < 0) 115270ecce91SMarcel Holtmann id = 0xffffffff; 115370ecce91SMarcel Holtmann 115470ecce91SMarcel Holtmann hci_pi(sk)->cookie = id; 115570ecce91SMarcel Holtmann get_task_comm(hci_pi(sk)->comm, current); 115670ecce91SMarcel Holtmann 1157249fa169SMarcel Holtmann /* Send event to monitor */ 1158249fa169SMarcel Holtmann skb = create_monitor_ctrl_open(sk); 1159249fa169SMarcel Holtmann if (skb) { 1160249fa169SMarcel Holtmann hci_send_to_channel(HCI_CHANNEL_MONITOR, skb, 1161249fa169SMarcel Holtmann HCI_SOCK_TRUSTED, NULL); 1162249fa169SMarcel Holtmann kfree_skb(skb); 1163249fa169SMarcel Holtmann } 1164249fa169SMarcel Holtmann 1165f9207338SMarcel Holtmann hci_sock_set_flag(sk, HCI_MGMT_INDEX_EVENTS); 1166f9207338SMarcel Holtmann hci_sock_set_flag(sk, HCI_MGMT_UNCONF_INDEX_EVENTS); 1167f6b7712eSMarcel Holtmann hci_sock_set_flag(sk, HCI_MGMT_GENERIC_EVENTS); 1168f9207338SMarcel Holtmann } 1169801c1e8dSJohan Hedberg break; 1170801c1e8dSJohan Hedberg } 1171801c1e8dSJohan Hedberg 11727cc2ade2SMarcel Holtmann 11737cc2ade2SMarcel Holtmann hci_pi(sk)->channel = haddr.hci_channel; 11741da177e4SLinus Torvalds sk->sk_state = BT_BOUND; 11751da177e4SLinus Torvalds 11761da177e4SLinus Torvalds done: 11771da177e4SLinus Torvalds release_sock(sk); 11781da177e4SLinus Torvalds return err; 11791da177e4SLinus Torvalds } 11801da177e4SLinus Torvalds 11818fc9ced3SGustavo Padovan static int hci_sock_getname(struct socket *sock, struct sockaddr *addr, 11828fc9ced3SGustavo Padovan int *addr_len, int peer) 11831da177e4SLinus Torvalds { 11841da177e4SLinus Torvalds struct sockaddr_hci *haddr = (struct sockaddr_hci *)addr; 11851da177e4SLinus Torvalds struct sock *sk = sock->sk; 11869d4b68b2SMarcel Holtmann struct hci_dev *hdev; 11879d4b68b2SMarcel Holtmann int err = 0; 11881da177e4SLinus Torvalds 11891da177e4SLinus Torvalds BT_DBG("sock %p sk %p", sock, sk); 11901da177e4SLinus Torvalds 119106f43cbcSMarcel Holtmann if (peer) 119206f43cbcSMarcel Holtmann return -EOPNOTSUPP; 119306f43cbcSMarcel Holtmann 11941da177e4SLinus Torvalds lock_sock(sk); 11951da177e4SLinus Torvalds 11969d4b68b2SMarcel Holtmann hdev = hci_pi(sk)->hdev; 11979d4b68b2SMarcel Holtmann if (!hdev) { 11989d4b68b2SMarcel Holtmann err = -EBADFD; 11999d4b68b2SMarcel Holtmann goto done; 12009d4b68b2SMarcel Holtmann } 12019d4b68b2SMarcel Holtmann 12021da177e4SLinus Torvalds *addr_len = sizeof(*haddr); 12031da177e4SLinus Torvalds haddr->hci_family = AF_BLUETOOTH; 12047b005bd3SMarcel Holtmann haddr->hci_dev = hdev->id; 12059d4b68b2SMarcel Holtmann haddr->hci_channel= hci_pi(sk)->channel; 12061da177e4SLinus Torvalds 12079d4b68b2SMarcel Holtmann done: 12081da177e4SLinus Torvalds release_sock(sk); 12099d4b68b2SMarcel Holtmann return err; 12101da177e4SLinus Torvalds } 12111da177e4SLinus Torvalds 12126039aa73SGustavo Padovan static void hci_sock_cmsg(struct sock *sk, struct msghdr *msg, 12136039aa73SGustavo Padovan struct sk_buff *skb) 12141da177e4SLinus Torvalds { 12151da177e4SLinus Torvalds __u32 mask = hci_pi(sk)->cmsg_mask; 12161da177e4SLinus Torvalds 12170d48d939SMarcel Holtmann if (mask & HCI_CMSG_DIR) { 12180d48d939SMarcel Holtmann int incoming = bt_cb(skb)->incoming; 12198fc9ced3SGustavo Padovan put_cmsg(msg, SOL_HCI, HCI_CMSG_DIR, sizeof(incoming), 12208fc9ced3SGustavo Padovan &incoming); 12210d48d939SMarcel Holtmann } 12221da177e4SLinus Torvalds 1223a61bbcf2SPatrick McHardy if (mask & HCI_CMSG_TSTAMP) { 1224f6e623a6SJohann Felix Soden #ifdef CONFIG_COMPAT 1225f6e623a6SJohann Felix Soden struct compat_timeval ctv; 1226f6e623a6SJohann Felix Soden #endif 1227a61bbcf2SPatrick McHardy struct timeval tv; 1228767c5eb5SMarcel Holtmann void *data; 1229767c5eb5SMarcel Holtmann int len; 1230a61bbcf2SPatrick McHardy 1231a61bbcf2SPatrick McHardy skb_get_timestamp(skb, &tv); 1232767c5eb5SMarcel Holtmann 12331da97f83SDavid S. Miller data = &tv; 12341da97f83SDavid S. Miller len = sizeof(tv); 12351da97f83SDavid S. Miller #ifdef CONFIG_COMPAT 1236da88cea1SH. J. Lu if (!COMPAT_USE_64BIT_TIME && 1237da88cea1SH. J. Lu (msg->msg_flags & MSG_CMSG_COMPAT)) { 1238767c5eb5SMarcel Holtmann ctv.tv_sec = tv.tv_sec; 1239767c5eb5SMarcel Holtmann ctv.tv_usec = tv.tv_usec; 1240767c5eb5SMarcel Holtmann data = &ctv; 1241767c5eb5SMarcel Holtmann len = sizeof(ctv); 1242767c5eb5SMarcel Holtmann } 12431da97f83SDavid S. Miller #endif 1244767c5eb5SMarcel Holtmann 1245767c5eb5SMarcel Holtmann put_cmsg(msg, SOL_HCI, HCI_CMSG_TSTAMP, len, data); 1246a61bbcf2SPatrick McHardy } 12471da177e4SLinus Torvalds } 12481da177e4SLinus Torvalds 12498528d3f7SMarcel Holtmann static int hci_sock_recvmsg(struct socket *sock, struct msghdr *msg, 12508528d3f7SMarcel Holtmann size_t len, int flags) 12511da177e4SLinus Torvalds { 12521da177e4SLinus Torvalds int noblock = flags & MSG_DONTWAIT; 12531da177e4SLinus Torvalds struct sock *sk = sock->sk; 12541da177e4SLinus Torvalds struct sk_buff *skb; 12551da177e4SLinus Torvalds int copied, err; 125683871f8cSDenis Kenzior unsigned int skblen; 12571da177e4SLinus Torvalds 12581da177e4SLinus Torvalds BT_DBG("sock %p, sk %p", sock, sk); 12591da177e4SLinus Torvalds 1260d94a6104SMarcel Holtmann if (flags & MSG_OOB) 12611da177e4SLinus Torvalds return -EOPNOTSUPP; 12621da177e4SLinus Torvalds 1263ac714949SMarcel Holtmann if (hci_pi(sk)->channel == HCI_CHANNEL_LOGGING) 1264ac714949SMarcel Holtmann return -EOPNOTSUPP; 1265ac714949SMarcel Holtmann 12661da177e4SLinus Torvalds if (sk->sk_state == BT_CLOSED) 12671da177e4SLinus Torvalds return 0; 12681da177e4SLinus Torvalds 126970f23020SAndrei Emeltchenko skb = skb_recv_datagram(sk, flags, noblock, &err); 127070f23020SAndrei Emeltchenko if (!skb) 12711da177e4SLinus Torvalds return err; 12721da177e4SLinus Torvalds 127383871f8cSDenis Kenzior skblen = skb->len; 12741da177e4SLinus Torvalds copied = skb->len; 12751da177e4SLinus Torvalds if (len < copied) { 12761da177e4SLinus Torvalds msg->msg_flags |= MSG_TRUNC; 12771da177e4SLinus Torvalds copied = len; 12781da177e4SLinus Torvalds } 12791da177e4SLinus Torvalds 1280badff6d0SArnaldo Carvalho de Melo skb_reset_transport_header(skb); 128151f3d02bSDavid S. Miller err = skb_copy_datagram_msg(skb, 0, msg, copied); 12821da177e4SLinus Torvalds 12833a208627SMarcel Holtmann switch (hci_pi(sk)->channel) { 12843a208627SMarcel Holtmann case HCI_CHANNEL_RAW: 12851da177e4SLinus Torvalds hci_sock_cmsg(sk, msg, skb); 12863a208627SMarcel Holtmann break; 128723500189SMarcel Holtmann case HCI_CHANNEL_USER: 1288cd82e61cSMarcel Holtmann case HCI_CHANNEL_MONITOR: 1289cd82e61cSMarcel Holtmann sock_recv_timestamp(msg, sk, skb); 1290cd82e61cSMarcel Holtmann break; 1291801c1e8dSJohan Hedberg default: 1292801c1e8dSJohan Hedberg if (hci_mgmt_chan_find(hci_pi(sk)->channel)) 1293801c1e8dSJohan Hedberg sock_recv_timestamp(msg, sk, skb); 1294801c1e8dSJohan Hedberg break; 12953a208627SMarcel Holtmann } 12961da177e4SLinus Torvalds 12971da177e4SLinus Torvalds skb_free_datagram(sk, skb); 12981da177e4SLinus Torvalds 12994f34228bSLuiz Augusto von Dentz if (flags & MSG_TRUNC) 130083871f8cSDenis Kenzior copied = skblen; 130183871f8cSDenis Kenzior 13021da177e4SLinus Torvalds return err ? : copied; 13031da177e4SLinus Torvalds } 13041da177e4SLinus Torvalds 1305fa4335d7SJohan Hedberg static int hci_mgmt_cmd(struct hci_mgmt_chan *chan, struct sock *sk, 1306fa4335d7SJohan Hedberg struct msghdr *msg, size_t msglen) 1307fa4335d7SJohan Hedberg { 1308fa4335d7SJohan Hedberg void *buf; 1309fa4335d7SJohan Hedberg u8 *cp; 1310fa4335d7SJohan Hedberg struct mgmt_hdr *hdr; 1311fa4335d7SJohan Hedberg u16 opcode, index, len; 1312fa4335d7SJohan Hedberg struct hci_dev *hdev = NULL; 1313fa4335d7SJohan Hedberg const struct hci_mgmt_handler *handler; 1314fa4335d7SJohan Hedberg bool var_len, no_hdev; 1315fa4335d7SJohan Hedberg int err; 1316fa4335d7SJohan Hedberg 1317fa4335d7SJohan Hedberg BT_DBG("got %zu bytes", msglen); 1318fa4335d7SJohan Hedberg 1319fa4335d7SJohan Hedberg if (msglen < sizeof(*hdr)) 1320fa4335d7SJohan Hedberg return -EINVAL; 1321fa4335d7SJohan Hedberg 1322fa4335d7SJohan Hedberg buf = kmalloc(msglen, GFP_KERNEL); 1323fa4335d7SJohan Hedberg if (!buf) 1324fa4335d7SJohan Hedberg return -ENOMEM; 1325fa4335d7SJohan Hedberg 1326fa4335d7SJohan Hedberg if (memcpy_from_msg(buf, msg, msglen)) { 1327fa4335d7SJohan Hedberg err = -EFAULT; 1328fa4335d7SJohan Hedberg goto done; 1329fa4335d7SJohan Hedberg } 1330fa4335d7SJohan Hedberg 1331fa4335d7SJohan Hedberg hdr = buf; 1332fa4335d7SJohan Hedberg opcode = __le16_to_cpu(hdr->opcode); 1333fa4335d7SJohan Hedberg index = __le16_to_cpu(hdr->index); 1334fa4335d7SJohan Hedberg len = __le16_to_cpu(hdr->len); 1335fa4335d7SJohan Hedberg 1336fa4335d7SJohan Hedberg if (len != msglen - sizeof(*hdr)) { 1337fa4335d7SJohan Hedberg err = -EINVAL; 1338fa4335d7SJohan Hedberg goto done; 1339fa4335d7SJohan Hedberg } 1340fa4335d7SJohan Hedberg 1341*38ceaa00SMarcel Holtmann if (chan->channel == HCI_CHANNEL_CONTROL) { 1342*38ceaa00SMarcel Holtmann struct sk_buff *skb; 1343*38ceaa00SMarcel Holtmann 1344*38ceaa00SMarcel Holtmann /* Send event to monitor */ 1345*38ceaa00SMarcel Holtmann skb = create_monitor_ctrl_command(sk, index, opcode, len, 1346*38ceaa00SMarcel Holtmann buf + sizeof(*hdr)); 1347*38ceaa00SMarcel Holtmann if (skb) { 1348*38ceaa00SMarcel Holtmann hci_send_to_channel(HCI_CHANNEL_MONITOR, skb, 1349*38ceaa00SMarcel Holtmann HCI_SOCK_TRUSTED, NULL); 1350*38ceaa00SMarcel Holtmann kfree_skb(skb); 1351*38ceaa00SMarcel Holtmann } 1352*38ceaa00SMarcel Holtmann } 1353*38ceaa00SMarcel Holtmann 1354fa4335d7SJohan Hedberg if (opcode >= chan->handler_count || 1355fa4335d7SJohan Hedberg chan->handlers[opcode].func == NULL) { 1356fa4335d7SJohan Hedberg BT_DBG("Unknown op %u", opcode); 1357fa4335d7SJohan Hedberg err = mgmt_cmd_status(sk, index, opcode, 1358fa4335d7SJohan Hedberg MGMT_STATUS_UNKNOWN_COMMAND); 1359fa4335d7SJohan Hedberg goto done; 1360fa4335d7SJohan Hedberg } 1361fa4335d7SJohan Hedberg 1362fa4335d7SJohan Hedberg handler = &chan->handlers[opcode]; 1363fa4335d7SJohan Hedberg 1364fa4335d7SJohan Hedberg if (!hci_sock_test_flag(sk, HCI_SOCK_TRUSTED) && 1365fa4335d7SJohan Hedberg !(handler->flags & HCI_MGMT_UNTRUSTED)) { 1366fa4335d7SJohan Hedberg err = mgmt_cmd_status(sk, index, opcode, 1367fa4335d7SJohan Hedberg MGMT_STATUS_PERMISSION_DENIED); 1368fa4335d7SJohan Hedberg goto done; 1369fa4335d7SJohan Hedberg } 1370fa4335d7SJohan Hedberg 1371fa4335d7SJohan Hedberg if (index != MGMT_INDEX_NONE) { 1372fa4335d7SJohan Hedberg hdev = hci_dev_get(index); 1373fa4335d7SJohan Hedberg if (!hdev) { 1374fa4335d7SJohan Hedberg err = mgmt_cmd_status(sk, index, opcode, 1375fa4335d7SJohan Hedberg MGMT_STATUS_INVALID_INDEX); 1376fa4335d7SJohan Hedberg goto done; 1377fa4335d7SJohan Hedberg } 1378fa4335d7SJohan Hedberg 1379fa4335d7SJohan Hedberg if (hci_dev_test_flag(hdev, HCI_SETUP) || 1380fa4335d7SJohan Hedberg hci_dev_test_flag(hdev, HCI_CONFIG) || 1381fa4335d7SJohan Hedberg hci_dev_test_flag(hdev, HCI_USER_CHANNEL)) { 1382fa4335d7SJohan Hedberg err = mgmt_cmd_status(sk, index, opcode, 1383fa4335d7SJohan Hedberg MGMT_STATUS_INVALID_INDEX); 1384fa4335d7SJohan Hedberg goto done; 1385fa4335d7SJohan Hedberg } 1386fa4335d7SJohan Hedberg 1387fa4335d7SJohan Hedberg if (hci_dev_test_flag(hdev, HCI_UNCONFIGURED) && 1388fa4335d7SJohan Hedberg !(handler->flags & HCI_MGMT_UNCONFIGURED)) { 1389fa4335d7SJohan Hedberg err = mgmt_cmd_status(sk, index, opcode, 1390fa4335d7SJohan Hedberg MGMT_STATUS_INVALID_INDEX); 1391fa4335d7SJohan Hedberg goto done; 1392fa4335d7SJohan Hedberg } 1393fa4335d7SJohan Hedberg } 1394fa4335d7SJohan Hedberg 1395fa4335d7SJohan Hedberg no_hdev = (handler->flags & HCI_MGMT_NO_HDEV); 1396fa4335d7SJohan Hedberg if (no_hdev != !hdev) { 1397fa4335d7SJohan Hedberg err = mgmt_cmd_status(sk, index, opcode, 1398fa4335d7SJohan Hedberg MGMT_STATUS_INVALID_INDEX); 1399fa4335d7SJohan Hedberg goto done; 1400fa4335d7SJohan Hedberg } 1401fa4335d7SJohan Hedberg 1402fa4335d7SJohan Hedberg var_len = (handler->flags & HCI_MGMT_VAR_LEN); 1403fa4335d7SJohan Hedberg if ((var_len && len < handler->data_len) || 1404fa4335d7SJohan Hedberg (!var_len && len != handler->data_len)) { 1405fa4335d7SJohan Hedberg err = mgmt_cmd_status(sk, index, opcode, 1406fa4335d7SJohan Hedberg MGMT_STATUS_INVALID_PARAMS); 1407fa4335d7SJohan Hedberg goto done; 1408fa4335d7SJohan Hedberg } 1409fa4335d7SJohan Hedberg 1410fa4335d7SJohan Hedberg if (hdev && chan->hdev_init) 1411fa4335d7SJohan Hedberg chan->hdev_init(sk, hdev); 1412fa4335d7SJohan Hedberg 1413fa4335d7SJohan Hedberg cp = buf + sizeof(*hdr); 1414fa4335d7SJohan Hedberg 1415fa4335d7SJohan Hedberg err = handler->func(sk, hdev, cp, len); 1416fa4335d7SJohan Hedberg if (err < 0) 1417fa4335d7SJohan Hedberg goto done; 1418fa4335d7SJohan Hedberg 1419fa4335d7SJohan Hedberg err = msglen; 1420fa4335d7SJohan Hedberg 1421fa4335d7SJohan Hedberg done: 1422fa4335d7SJohan Hedberg if (hdev) 1423fa4335d7SJohan Hedberg hci_dev_put(hdev); 1424fa4335d7SJohan Hedberg 1425fa4335d7SJohan Hedberg kfree(buf); 1426fa4335d7SJohan Hedberg return err; 1427fa4335d7SJohan Hedberg } 1428fa4335d7SJohan Hedberg 1429ac714949SMarcel Holtmann static int hci_logging_frame(struct sock *sk, struct msghdr *msg, int len) 1430ac714949SMarcel Holtmann { 1431ac714949SMarcel Holtmann struct hci_mon_hdr *hdr; 1432ac714949SMarcel Holtmann struct sk_buff *skb; 1433ac714949SMarcel Holtmann struct hci_dev *hdev; 1434ac714949SMarcel Holtmann u16 index; 1435ac714949SMarcel Holtmann int err; 1436ac714949SMarcel Holtmann 1437ac714949SMarcel Holtmann /* The logging frame consists at minimum of the standard header, 1438ac714949SMarcel Holtmann * the priority byte, the ident length byte and at least one string 1439ac714949SMarcel Holtmann * terminator NUL byte. Anything shorter are invalid packets. 1440ac714949SMarcel Holtmann */ 1441ac714949SMarcel Holtmann if (len < sizeof(*hdr) + 3) 1442ac714949SMarcel Holtmann return -EINVAL; 1443ac714949SMarcel Holtmann 1444ac714949SMarcel Holtmann skb = bt_skb_send_alloc(sk, len, msg->msg_flags & MSG_DONTWAIT, &err); 1445ac714949SMarcel Holtmann if (!skb) 1446ac714949SMarcel Holtmann return err; 1447ac714949SMarcel Holtmann 1448ac714949SMarcel Holtmann if (memcpy_from_msg(skb_put(skb, len), msg, len)) { 1449ac714949SMarcel Holtmann err = -EFAULT; 1450ac714949SMarcel Holtmann goto drop; 1451ac714949SMarcel Holtmann } 1452ac714949SMarcel Holtmann 1453ac714949SMarcel Holtmann hdr = (void *)skb->data; 1454ac714949SMarcel Holtmann 1455ac714949SMarcel Holtmann if (__le16_to_cpu(hdr->len) != len - sizeof(*hdr)) { 1456ac714949SMarcel Holtmann err = -EINVAL; 1457ac714949SMarcel Holtmann goto drop; 1458ac714949SMarcel Holtmann } 1459ac714949SMarcel Holtmann 1460ac714949SMarcel Holtmann if (__le16_to_cpu(hdr->opcode) == 0x0000) { 1461ac714949SMarcel Holtmann __u8 priority = skb->data[sizeof(*hdr)]; 1462ac714949SMarcel Holtmann __u8 ident_len = skb->data[sizeof(*hdr) + 1]; 1463ac714949SMarcel Holtmann 1464ac714949SMarcel Holtmann /* Only the priorities 0-7 are valid and with that any other 1465ac714949SMarcel Holtmann * value results in an invalid packet. 1466ac714949SMarcel Holtmann * 1467ac714949SMarcel Holtmann * The priority byte is followed by an ident length byte and 1468ac714949SMarcel Holtmann * the NUL terminated ident string. Check that the ident 1469ac714949SMarcel Holtmann * length is not overflowing the packet and also that the 1470ac714949SMarcel Holtmann * ident string itself is NUL terminated. In case the ident 1471ac714949SMarcel Holtmann * length is zero, the length value actually doubles as NUL 1472ac714949SMarcel Holtmann * terminator identifier. 1473ac714949SMarcel Holtmann * 1474ac714949SMarcel Holtmann * The message follows the ident string (if present) and 1475ac714949SMarcel Holtmann * must be NUL terminated. Otherwise it is not a valid packet. 1476ac714949SMarcel Holtmann */ 1477ac714949SMarcel Holtmann if (priority > 7 || skb->data[len - 1] != 0x00 || 1478ac714949SMarcel Holtmann ident_len > len - sizeof(*hdr) - 3 || 1479ac714949SMarcel Holtmann skb->data[sizeof(*hdr) + ident_len + 1] != 0x00) { 1480ac714949SMarcel Holtmann err = -EINVAL; 1481ac714949SMarcel Holtmann goto drop; 1482ac714949SMarcel Holtmann } 1483ac714949SMarcel Holtmann } else { 1484ac714949SMarcel Holtmann err = -EINVAL; 1485ac714949SMarcel Holtmann goto drop; 1486ac714949SMarcel Holtmann } 1487ac714949SMarcel Holtmann 1488ac714949SMarcel Holtmann index = __le16_to_cpu(hdr->index); 1489ac714949SMarcel Holtmann 1490ac714949SMarcel Holtmann if (index != MGMT_INDEX_NONE) { 1491ac714949SMarcel Holtmann hdev = hci_dev_get(index); 1492ac714949SMarcel Holtmann if (!hdev) { 1493ac714949SMarcel Holtmann err = -ENODEV; 1494ac714949SMarcel Holtmann goto drop; 1495ac714949SMarcel Holtmann } 1496ac714949SMarcel Holtmann } else { 1497ac714949SMarcel Holtmann hdev = NULL; 1498ac714949SMarcel Holtmann } 1499ac714949SMarcel Holtmann 1500ac714949SMarcel Holtmann hdr->opcode = cpu_to_le16(HCI_MON_USER_LOGGING); 1501ac714949SMarcel Holtmann 1502ac714949SMarcel Holtmann hci_send_to_channel(HCI_CHANNEL_MONITOR, skb, HCI_SOCK_TRUSTED, NULL); 1503ac714949SMarcel Holtmann err = len; 1504ac714949SMarcel Holtmann 1505ac714949SMarcel Holtmann if (hdev) 1506ac714949SMarcel Holtmann hci_dev_put(hdev); 1507ac714949SMarcel Holtmann 1508ac714949SMarcel Holtmann drop: 1509ac714949SMarcel Holtmann kfree_skb(skb); 1510ac714949SMarcel Holtmann return err; 1511ac714949SMarcel Holtmann } 1512ac714949SMarcel Holtmann 15131b784140SYing Xue static int hci_sock_sendmsg(struct socket *sock, struct msghdr *msg, 15141b784140SYing Xue size_t len) 15151da177e4SLinus Torvalds { 15161da177e4SLinus Torvalds struct sock *sk = sock->sk; 1517801c1e8dSJohan Hedberg struct hci_mgmt_chan *chan; 15181da177e4SLinus Torvalds struct hci_dev *hdev; 15191da177e4SLinus Torvalds struct sk_buff *skb; 15201da177e4SLinus Torvalds int err; 15211da177e4SLinus Torvalds 15221da177e4SLinus Torvalds BT_DBG("sock %p sk %p", sock, sk); 15231da177e4SLinus Torvalds 15241da177e4SLinus Torvalds if (msg->msg_flags & MSG_OOB) 15251da177e4SLinus Torvalds return -EOPNOTSUPP; 15261da177e4SLinus Torvalds 15271da177e4SLinus Torvalds if (msg->msg_flags & ~(MSG_DONTWAIT|MSG_NOSIGNAL|MSG_ERRQUEUE)) 15281da177e4SLinus Torvalds return -EINVAL; 15291da177e4SLinus Torvalds 15301da177e4SLinus Torvalds if (len < 4 || len > HCI_MAX_FRAME_SIZE) 15311da177e4SLinus Torvalds return -EINVAL; 15321da177e4SLinus Torvalds 15331da177e4SLinus Torvalds lock_sock(sk); 15341da177e4SLinus Torvalds 15350381101fSJohan Hedberg switch (hci_pi(sk)->channel) { 15360381101fSJohan Hedberg case HCI_CHANNEL_RAW: 153723500189SMarcel Holtmann case HCI_CHANNEL_USER: 15380381101fSJohan Hedberg break; 1539cd82e61cSMarcel Holtmann case HCI_CHANNEL_MONITOR: 1540cd82e61cSMarcel Holtmann err = -EOPNOTSUPP; 1541cd82e61cSMarcel Holtmann goto done; 1542ac714949SMarcel Holtmann case HCI_CHANNEL_LOGGING: 1543ac714949SMarcel Holtmann err = hci_logging_frame(sk, msg, len); 1544ac714949SMarcel Holtmann goto done; 15450381101fSJohan Hedberg default: 1546801c1e8dSJohan Hedberg mutex_lock(&mgmt_chan_list_lock); 1547801c1e8dSJohan Hedberg chan = __hci_mgmt_chan_find(hci_pi(sk)->channel); 1548801c1e8dSJohan Hedberg if (chan) 1549fa4335d7SJohan Hedberg err = hci_mgmt_cmd(chan, sk, msg, len); 1550801c1e8dSJohan Hedberg else 15510381101fSJohan Hedberg err = -EINVAL; 1552801c1e8dSJohan Hedberg 1553801c1e8dSJohan Hedberg mutex_unlock(&mgmt_chan_list_lock); 15540381101fSJohan Hedberg goto done; 15550381101fSJohan Hedberg } 15560381101fSJohan Hedberg 155770f23020SAndrei Emeltchenko hdev = hci_pi(sk)->hdev; 155870f23020SAndrei Emeltchenko if (!hdev) { 15591da177e4SLinus Torvalds err = -EBADFD; 15601da177e4SLinus Torvalds goto done; 15611da177e4SLinus Torvalds } 15621da177e4SLinus Torvalds 15637e21addcSMarcel Holtmann if (!test_bit(HCI_UP, &hdev->flags)) { 15647e21addcSMarcel Holtmann err = -ENETDOWN; 15657e21addcSMarcel Holtmann goto done; 15667e21addcSMarcel Holtmann } 15677e21addcSMarcel Holtmann 156870f23020SAndrei Emeltchenko skb = bt_skb_send_alloc(sk, len, msg->msg_flags & MSG_DONTWAIT, &err); 156970f23020SAndrei Emeltchenko if (!skb) 15701da177e4SLinus Torvalds goto done; 15711da177e4SLinus Torvalds 15726ce8e9ceSAl Viro if (memcpy_from_msg(skb_put(skb, len), msg, len)) { 15731da177e4SLinus Torvalds err = -EFAULT; 15741da177e4SLinus Torvalds goto drop; 15751da177e4SLinus Torvalds } 15761da177e4SLinus Torvalds 15778528d3f7SMarcel Holtmann hci_skb_pkt_type(skb) = skb->data[0]; 15781da177e4SLinus Torvalds skb_pull(skb, 1); 15791da177e4SLinus Torvalds 15801bc5ad16SMarcel Holtmann if (hci_pi(sk)->channel == HCI_CHANNEL_USER) { 15811bc5ad16SMarcel Holtmann /* No permission check is needed for user channel 15821bc5ad16SMarcel Holtmann * since that gets enforced when binding the socket. 15831bc5ad16SMarcel Holtmann * 15841bc5ad16SMarcel Holtmann * However check that the packet type is valid. 15851bc5ad16SMarcel Holtmann */ 1586d79f34e3SMarcel Holtmann if (hci_skb_pkt_type(skb) != HCI_COMMAND_PKT && 1587d79f34e3SMarcel Holtmann hci_skb_pkt_type(skb) != HCI_ACLDATA_PKT && 1588d79f34e3SMarcel Holtmann hci_skb_pkt_type(skb) != HCI_SCODATA_PKT) { 15891bc5ad16SMarcel Holtmann err = -EINVAL; 15901bc5ad16SMarcel Holtmann goto drop; 15911bc5ad16SMarcel Holtmann } 15921bc5ad16SMarcel Holtmann 15931bc5ad16SMarcel Holtmann skb_queue_tail(&hdev->raw_q, skb); 15941bc5ad16SMarcel Holtmann queue_work(hdev->workqueue, &hdev->tx_work); 1595d79f34e3SMarcel Holtmann } else if (hci_skb_pkt_type(skb) == HCI_COMMAND_PKT) { 159683985319SHarvey Harrison u16 opcode = get_unaligned_le16(skb->data); 15971da177e4SLinus Torvalds u16 ogf = hci_opcode_ogf(opcode); 15981da177e4SLinus Torvalds u16 ocf = hci_opcode_ocf(opcode); 15991da177e4SLinus Torvalds 16001da177e4SLinus Torvalds if (((ogf > HCI_SFLT_MAX_OGF) || 16013bb3c755SGustavo Padovan !hci_test_bit(ocf & HCI_FLT_OCF_BITS, 16023bb3c755SGustavo Padovan &hci_sec_filter.ocf_mask[ogf])) && 16031da177e4SLinus Torvalds !capable(CAP_NET_RAW)) { 16041da177e4SLinus Torvalds err = -EPERM; 16051da177e4SLinus Torvalds goto drop; 16061da177e4SLinus Torvalds } 16071da177e4SLinus Torvalds 16081982162bSMarcel Holtmann /* Since the opcode has already been extracted here, store 16091982162bSMarcel Holtmann * a copy of the value for later use by the drivers. 16101982162bSMarcel Holtmann */ 16111982162bSMarcel Holtmann hci_skb_opcode(skb) = opcode; 16121982162bSMarcel Holtmann 1613fee746b0SMarcel Holtmann if (ogf == 0x3f) { 16141da177e4SLinus Torvalds skb_queue_tail(&hdev->raw_q, skb); 16153eff45eaSGustavo F. Padovan queue_work(hdev->workqueue, &hdev->tx_work); 16161da177e4SLinus Torvalds } else { 161749c922bbSStephen Hemminger /* Stand-alone HCI commands must be flagged as 161811714b3dSJohan Hedberg * single-command requests. 161911714b3dSJohan Hedberg */ 162044d27137SJohan Hedberg bt_cb(skb)->hci.req_flags |= HCI_REQ_START; 162111714b3dSJohan Hedberg 16221da177e4SLinus Torvalds skb_queue_tail(&hdev->cmd_q, skb); 1623c347b765SGustavo F. Padovan queue_work(hdev->workqueue, &hdev->cmd_work); 16241da177e4SLinus Torvalds } 16251da177e4SLinus Torvalds } else { 16261da177e4SLinus Torvalds if (!capable(CAP_NET_RAW)) { 16271da177e4SLinus Torvalds err = -EPERM; 16281da177e4SLinus Torvalds goto drop; 16291da177e4SLinus Torvalds } 16301da177e4SLinus Torvalds 1631d79f34e3SMarcel Holtmann if (hci_skb_pkt_type(skb) != HCI_ACLDATA_PKT && 1632d79f34e3SMarcel Holtmann hci_skb_pkt_type(skb) != HCI_SCODATA_PKT) { 1633bb77543eSMarcel Holtmann err = -EINVAL; 1634bb77543eSMarcel Holtmann goto drop; 1635bb77543eSMarcel Holtmann } 1636bb77543eSMarcel Holtmann 16371da177e4SLinus Torvalds skb_queue_tail(&hdev->raw_q, skb); 16383eff45eaSGustavo F. Padovan queue_work(hdev->workqueue, &hdev->tx_work); 16391da177e4SLinus Torvalds } 16401da177e4SLinus Torvalds 16411da177e4SLinus Torvalds err = len; 16421da177e4SLinus Torvalds 16431da177e4SLinus Torvalds done: 16441da177e4SLinus Torvalds release_sock(sk); 16451da177e4SLinus Torvalds return err; 16461da177e4SLinus Torvalds 16471da177e4SLinus Torvalds drop: 16481da177e4SLinus Torvalds kfree_skb(skb); 16491da177e4SLinus Torvalds goto done; 16501da177e4SLinus Torvalds } 16511da177e4SLinus Torvalds 16528fc9ced3SGustavo Padovan static int hci_sock_setsockopt(struct socket *sock, int level, int optname, 16538fc9ced3SGustavo Padovan char __user *optval, unsigned int len) 16541da177e4SLinus Torvalds { 16551da177e4SLinus Torvalds struct hci_ufilter uf = { .opcode = 0 }; 16561da177e4SLinus Torvalds struct sock *sk = sock->sk; 16571da177e4SLinus Torvalds int err = 0, opt = 0; 16581da177e4SLinus Torvalds 16591da177e4SLinus Torvalds BT_DBG("sk %p, opt %d", sk, optname); 16601da177e4SLinus Torvalds 166147b0f573SMarcel Holtmann if (level != SOL_HCI) 166247b0f573SMarcel Holtmann return -ENOPROTOOPT; 166347b0f573SMarcel Holtmann 16641da177e4SLinus Torvalds lock_sock(sk); 16651da177e4SLinus Torvalds 16662f39cdb7SMarcel Holtmann if (hci_pi(sk)->channel != HCI_CHANNEL_RAW) { 1667c2371e80SMarcel Holtmann err = -EBADFD; 16682f39cdb7SMarcel Holtmann goto done; 16692f39cdb7SMarcel Holtmann } 16702f39cdb7SMarcel Holtmann 16711da177e4SLinus Torvalds switch (optname) { 16721da177e4SLinus Torvalds case HCI_DATA_DIR: 16731da177e4SLinus Torvalds if (get_user(opt, (int __user *)optval)) { 16741da177e4SLinus Torvalds err = -EFAULT; 16751da177e4SLinus Torvalds break; 16761da177e4SLinus Torvalds } 16771da177e4SLinus Torvalds 16781da177e4SLinus Torvalds if (opt) 16791da177e4SLinus Torvalds hci_pi(sk)->cmsg_mask |= HCI_CMSG_DIR; 16801da177e4SLinus Torvalds else 16811da177e4SLinus Torvalds hci_pi(sk)->cmsg_mask &= ~HCI_CMSG_DIR; 16821da177e4SLinus Torvalds break; 16831da177e4SLinus Torvalds 16841da177e4SLinus Torvalds case HCI_TIME_STAMP: 16851da177e4SLinus Torvalds if (get_user(opt, (int __user *)optval)) { 16861da177e4SLinus Torvalds err = -EFAULT; 16871da177e4SLinus Torvalds break; 16881da177e4SLinus Torvalds } 16891da177e4SLinus Torvalds 16901da177e4SLinus Torvalds if (opt) 16911da177e4SLinus Torvalds hci_pi(sk)->cmsg_mask |= HCI_CMSG_TSTAMP; 16921da177e4SLinus Torvalds else 16931da177e4SLinus Torvalds hci_pi(sk)->cmsg_mask &= ~HCI_CMSG_TSTAMP; 16941da177e4SLinus Torvalds break; 16951da177e4SLinus Torvalds 16961da177e4SLinus Torvalds case HCI_FILTER: 16970878b666SMarcel Holtmann { 16980878b666SMarcel Holtmann struct hci_filter *f = &hci_pi(sk)->filter; 16990878b666SMarcel Holtmann 17000878b666SMarcel Holtmann uf.type_mask = f->type_mask; 17010878b666SMarcel Holtmann uf.opcode = f->opcode; 17020878b666SMarcel Holtmann uf.event_mask[0] = *((u32 *) f->event_mask + 0); 17030878b666SMarcel Holtmann uf.event_mask[1] = *((u32 *) f->event_mask + 1); 17040878b666SMarcel Holtmann } 17050878b666SMarcel Holtmann 17061da177e4SLinus Torvalds len = min_t(unsigned int, len, sizeof(uf)); 17071da177e4SLinus Torvalds if (copy_from_user(&uf, optval, len)) { 17081da177e4SLinus Torvalds err = -EFAULT; 17091da177e4SLinus Torvalds break; 17101da177e4SLinus Torvalds } 17111da177e4SLinus Torvalds 17121da177e4SLinus Torvalds if (!capable(CAP_NET_RAW)) { 17131da177e4SLinus Torvalds uf.type_mask &= hci_sec_filter.type_mask; 17141da177e4SLinus Torvalds uf.event_mask[0] &= *((u32 *) hci_sec_filter.event_mask + 0); 17151da177e4SLinus Torvalds uf.event_mask[1] &= *((u32 *) hci_sec_filter.event_mask + 1); 17161da177e4SLinus Torvalds } 17171da177e4SLinus Torvalds 17181da177e4SLinus Torvalds { 17191da177e4SLinus Torvalds struct hci_filter *f = &hci_pi(sk)->filter; 17201da177e4SLinus Torvalds 17211da177e4SLinus Torvalds f->type_mask = uf.type_mask; 17221da177e4SLinus Torvalds f->opcode = uf.opcode; 17231da177e4SLinus Torvalds *((u32 *) f->event_mask + 0) = uf.event_mask[0]; 17241da177e4SLinus Torvalds *((u32 *) f->event_mask + 1) = uf.event_mask[1]; 17251da177e4SLinus Torvalds } 17261da177e4SLinus Torvalds break; 17271da177e4SLinus Torvalds 17281da177e4SLinus Torvalds default: 17291da177e4SLinus Torvalds err = -ENOPROTOOPT; 17301da177e4SLinus Torvalds break; 17311da177e4SLinus Torvalds } 17321da177e4SLinus Torvalds 17332f39cdb7SMarcel Holtmann done: 17341da177e4SLinus Torvalds release_sock(sk); 17351da177e4SLinus Torvalds return err; 17361da177e4SLinus Torvalds } 17371da177e4SLinus Torvalds 17388fc9ced3SGustavo Padovan static int hci_sock_getsockopt(struct socket *sock, int level, int optname, 17398fc9ced3SGustavo Padovan char __user *optval, int __user *optlen) 17401da177e4SLinus Torvalds { 17411da177e4SLinus Torvalds struct hci_ufilter uf; 17421da177e4SLinus Torvalds struct sock *sk = sock->sk; 1743cedc5469SMarcel Holtmann int len, opt, err = 0; 1744cedc5469SMarcel Holtmann 1745cedc5469SMarcel Holtmann BT_DBG("sk %p, opt %d", sk, optname); 17461da177e4SLinus Torvalds 174747b0f573SMarcel Holtmann if (level != SOL_HCI) 174847b0f573SMarcel Holtmann return -ENOPROTOOPT; 174947b0f573SMarcel Holtmann 17501da177e4SLinus Torvalds if (get_user(len, optlen)) 17511da177e4SLinus Torvalds return -EFAULT; 17521da177e4SLinus Torvalds 1753cedc5469SMarcel Holtmann lock_sock(sk); 1754cedc5469SMarcel Holtmann 1755cedc5469SMarcel Holtmann if (hci_pi(sk)->channel != HCI_CHANNEL_RAW) { 1756c2371e80SMarcel Holtmann err = -EBADFD; 1757cedc5469SMarcel Holtmann goto done; 1758cedc5469SMarcel Holtmann } 1759cedc5469SMarcel Holtmann 17601da177e4SLinus Torvalds switch (optname) { 17611da177e4SLinus Torvalds case HCI_DATA_DIR: 17621da177e4SLinus Torvalds if (hci_pi(sk)->cmsg_mask & HCI_CMSG_DIR) 17631da177e4SLinus Torvalds opt = 1; 17641da177e4SLinus Torvalds else 17651da177e4SLinus Torvalds opt = 0; 17661da177e4SLinus Torvalds 17671da177e4SLinus Torvalds if (put_user(opt, optval)) 1768cedc5469SMarcel Holtmann err = -EFAULT; 17691da177e4SLinus Torvalds break; 17701da177e4SLinus Torvalds 17711da177e4SLinus Torvalds case HCI_TIME_STAMP: 17721da177e4SLinus Torvalds if (hci_pi(sk)->cmsg_mask & HCI_CMSG_TSTAMP) 17731da177e4SLinus Torvalds opt = 1; 17741da177e4SLinus Torvalds else 17751da177e4SLinus Torvalds opt = 0; 17761da177e4SLinus Torvalds 17771da177e4SLinus Torvalds if (put_user(opt, optval)) 1778cedc5469SMarcel Holtmann err = -EFAULT; 17791da177e4SLinus Torvalds break; 17801da177e4SLinus Torvalds 17811da177e4SLinus Torvalds case HCI_FILTER: 17821da177e4SLinus Torvalds { 17831da177e4SLinus Torvalds struct hci_filter *f = &hci_pi(sk)->filter; 17841da177e4SLinus Torvalds 1785e15ca9a0SMathias Krause memset(&uf, 0, sizeof(uf)); 17861da177e4SLinus Torvalds uf.type_mask = f->type_mask; 17871da177e4SLinus Torvalds uf.opcode = f->opcode; 17881da177e4SLinus Torvalds uf.event_mask[0] = *((u32 *) f->event_mask + 0); 17891da177e4SLinus Torvalds uf.event_mask[1] = *((u32 *) f->event_mask + 1); 17901da177e4SLinus Torvalds } 17911da177e4SLinus Torvalds 17921da177e4SLinus Torvalds len = min_t(unsigned int, len, sizeof(uf)); 17931da177e4SLinus Torvalds if (copy_to_user(optval, &uf, len)) 1794cedc5469SMarcel Holtmann err = -EFAULT; 17951da177e4SLinus Torvalds break; 17961da177e4SLinus Torvalds 17971da177e4SLinus Torvalds default: 1798cedc5469SMarcel Holtmann err = -ENOPROTOOPT; 17991da177e4SLinus Torvalds break; 18001da177e4SLinus Torvalds } 18011da177e4SLinus Torvalds 1802cedc5469SMarcel Holtmann done: 1803cedc5469SMarcel Holtmann release_sock(sk); 1804cedc5469SMarcel Holtmann return err; 18051da177e4SLinus Torvalds } 18061da177e4SLinus Torvalds 180790ddc4f0SEric Dumazet static const struct proto_ops hci_sock_ops = { 18081da177e4SLinus Torvalds .family = PF_BLUETOOTH, 18091da177e4SLinus Torvalds .owner = THIS_MODULE, 18101da177e4SLinus Torvalds .release = hci_sock_release, 18111da177e4SLinus Torvalds .bind = hci_sock_bind, 18121da177e4SLinus Torvalds .getname = hci_sock_getname, 18131da177e4SLinus Torvalds .sendmsg = hci_sock_sendmsg, 18141da177e4SLinus Torvalds .recvmsg = hci_sock_recvmsg, 18151da177e4SLinus Torvalds .ioctl = hci_sock_ioctl, 18161da177e4SLinus Torvalds .poll = datagram_poll, 18171da177e4SLinus Torvalds .listen = sock_no_listen, 18181da177e4SLinus Torvalds .shutdown = sock_no_shutdown, 18191da177e4SLinus Torvalds .setsockopt = hci_sock_setsockopt, 18201da177e4SLinus Torvalds .getsockopt = hci_sock_getsockopt, 18211da177e4SLinus Torvalds .connect = sock_no_connect, 18221da177e4SLinus Torvalds .socketpair = sock_no_socketpair, 18231da177e4SLinus Torvalds .accept = sock_no_accept, 18241da177e4SLinus Torvalds .mmap = sock_no_mmap 18251da177e4SLinus Torvalds }; 18261da177e4SLinus Torvalds 18271da177e4SLinus Torvalds static struct proto hci_sk_proto = { 18281da177e4SLinus Torvalds .name = "HCI", 18291da177e4SLinus Torvalds .owner = THIS_MODULE, 18301da177e4SLinus Torvalds .obj_size = sizeof(struct hci_pinfo) 18311da177e4SLinus Torvalds }; 18321da177e4SLinus Torvalds 18333f378b68SEric Paris static int hci_sock_create(struct net *net, struct socket *sock, int protocol, 18343f378b68SEric Paris int kern) 18351da177e4SLinus Torvalds { 18361da177e4SLinus Torvalds struct sock *sk; 18371da177e4SLinus Torvalds 18381da177e4SLinus Torvalds BT_DBG("sock %p", sock); 18391da177e4SLinus Torvalds 18401da177e4SLinus Torvalds if (sock->type != SOCK_RAW) 18411da177e4SLinus Torvalds return -ESOCKTNOSUPPORT; 18421da177e4SLinus Torvalds 18431da177e4SLinus Torvalds sock->ops = &hci_sock_ops; 18441da177e4SLinus Torvalds 184511aa9c28SEric W. Biederman sk = sk_alloc(net, PF_BLUETOOTH, GFP_ATOMIC, &hci_sk_proto, kern); 18461da177e4SLinus Torvalds if (!sk) 18471da177e4SLinus Torvalds return -ENOMEM; 18481da177e4SLinus Torvalds 18491da177e4SLinus Torvalds sock_init_data(sock, sk); 18501da177e4SLinus Torvalds 18511da177e4SLinus Torvalds sock_reset_flag(sk, SOCK_ZAPPED); 18521da177e4SLinus Torvalds 18531da177e4SLinus Torvalds sk->sk_protocol = protocol; 18541da177e4SLinus Torvalds 18551da177e4SLinus Torvalds sock->state = SS_UNCONNECTED; 18561da177e4SLinus Torvalds sk->sk_state = BT_OPEN; 18571da177e4SLinus Torvalds 18581da177e4SLinus Torvalds bt_sock_link(&hci_sk_list, sk); 18591da177e4SLinus Torvalds return 0; 18601da177e4SLinus Torvalds } 18611da177e4SLinus Torvalds 1862ec1b4cf7SStephen Hemminger static const struct net_proto_family hci_sock_family_ops = { 18631da177e4SLinus Torvalds .family = PF_BLUETOOTH, 18641da177e4SLinus Torvalds .owner = THIS_MODULE, 18651da177e4SLinus Torvalds .create = hci_sock_create, 18661da177e4SLinus Torvalds }; 18671da177e4SLinus Torvalds 18681da177e4SLinus Torvalds int __init hci_sock_init(void) 18691da177e4SLinus Torvalds { 18701da177e4SLinus Torvalds int err; 18711da177e4SLinus Torvalds 1872b0a8e282SMarcel Holtmann BUILD_BUG_ON(sizeof(struct sockaddr_hci) > sizeof(struct sockaddr)); 1873b0a8e282SMarcel Holtmann 18741da177e4SLinus Torvalds err = proto_register(&hci_sk_proto, 0); 18751da177e4SLinus Torvalds if (err < 0) 18761da177e4SLinus Torvalds return err; 18771da177e4SLinus Torvalds 18781da177e4SLinus Torvalds err = bt_sock_register(BTPROTO_HCI, &hci_sock_family_ops); 1879f7c86637SMasatake YAMATO if (err < 0) { 1880f7c86637SMasatake YAMATO BT_ERR("HCI socket registration failed"); 18811da177e4SLinus Torvalds goto error; 1882f7c86637SMasatake YAMATO } 1883f7c86637SMasatake YAMATO 1884b0316615SAl Viro err = bt_procfs_init(&init_net, "hci", &hci_sk_list, NULL); 1885f7c86637SMasatake YAMATO if (err < 0) { 1886f7c86637SMasatake YAMATO BT_ERR("Failed to create HCI proc file"); 1887f7c86637SMasatake YAMATO bt_sock_unregister(BTPROTO_HCI); 1888f7c86637SMasatake YAMATO goto error; 1889f7c86637SMasatake YAMATO } 18901da177e4SLinus Torvalds 18911da177e4SLinus Torvalds BT_INFO("HCI socket layer initialized"); 18921da177e4SLinus Torvalds 18931da177e4SLinus Torvalds return 0; 18941da177e4SLinus Torvalds 18951da177e4SLinus Torvalds error: 18961da177e4SLinus Torvalds proto_unregister(&hci_sk_proto); 18971da177e4SLinus Torvalds return err; 18981da177e4SLinus Torvalds } 18991da177e4SLinus Torvalds 1900b7440a14SAnand Gadiyar void hci_sock_cleanup(void) 19011da177e4SLinus Torvalds { 1902f7c86637SMasatake YAMATO bt_procfs_cleanup(&init_net, "hci"); 19035e9d7f86SDavid Herrmann bt_sock_unregister(BTPROTO_HCI); 19041da177e4SLinus Torvalds proto_unregister(&hci_sk_proto); 19051da177e4SLinus Torvalds } 1906