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 87df1cb87aSMarcel Holtmann static bool hci_sock_gen_cookie(struct sock *sk) 88df1cb87aSMarcel Holtmann { 89df1cb87aSMarcel Holtmann int id = hci_pi(sk)->cookie; 90df1cb87aSMarcel Holtmann 91df1cb87aSMarcel Holtmann if (!id) { 92df1cb87aSMarcel Holtmann id = ida_simple_get(&sock_cookie_ida, 1, 0, GFP_KERNEL); 93df1cb87aSMarcel Holtmann if (id < 0) 94df1cb87aSMarcel Holtmann id = 0xffffffff; 95df1cb87aSMarcel Holtmann 96df1cb87aSMarcel Holtmann hci_pi(sk)->cookie = id; 97df1cb87aSMarcel Holtmann get_task_comm(hci_pi(sk)->comm, current); 98df1cb87aSMarcel Holtmann return true; 99df1cb87aSMarcel Holtmann } 100df1cb87aSMarcel Holtmann 101df1cb87aSMarcel Holtmann return false; 102df1cb87aSMarcel Holtmann } 103df1cb87aSMarcel Holtmann 104df1cb87aSMarcel Holtmann static void hci_sock_free_cookie(struct sock *sk) 105df1cb87aSMarcel Holtmann { 106df1cb87aSMarcel Holtmann int id = hci_pi(sk)->cookie; 107df1cb87aSMarcel Holtmann 108df1cb87aSMarcel Holtmann if (id) { 109df1cb87aSMarcel Holtmann hci_pi(sk)->cookie = 0xffffffff; 110df1cb87aSMarcel Holtmann ida_simple_remove(&sock_cookie_ida, id); 111df1cb87aSMarcel Holtmann } 112df1cb87aSMarcel Holtmann } 113df1cb87aSMarcel Holtmann 1149391976aSJiri Slaby static inline int hci_test_bit(int nr, const void *addr) 1151da177e4SLinus Torvalds { 1169391976aSJiri Slaby return *((const __u32 *) addr + (nr >> 5)) & ((__u32) 1 << (nr & 31)); 1171da177e4SLinus Torvalds } 1181da177e4SLinus Torvalds 1191da177e4SLinus Torvalds /* Security filter */ 1203ad254f7SMarcel Holtmann #define HCI_SFLT_MAX_OGF 5 1213ad254f7SMarcel Holtmann 1223ad254f7SMarcel Holtmann struct hci_sec_filter { 1233ad254f7SMarcel Holtmann __u32 type_mask; 1243ad254f7SMarcel Holtmann __u32 event_mask[2]; 1253ad254f7SMarcel Holtmann __u32 ocf_mask[HCI_SFLT_MAX_OGF + 1][4]; 1263ad254f7SMarcel Holtmann }; 1273ad254f7SMarcel Holtmann 1287e67c112SMarcel Holtmann static const struct hci_sec_filter hci_sec_filter = { 1291da177e4SLinus Torvalds /* Packet types */ 1301da177e4SLinus Torvalds 0x10, 1311da177e4SLinus Torvalds /* Events */ 132dd7f5527SMarcel Holtmann { 0x1000d9fe, 0x0000b00c }, 1331da177e4SLinus Torvalds /* Commands */ 1341da177e4SLinus Torvalds { 1351da177e4SLinus Torvalds { 0x0 }, 1361da177e4SLinus Torvalds /* OGF_LINK_CTL */ 1377c631a67SMarcel Holtmann { 0xbe000006, 0x00000001, 0x00000000, 0x00 }, 1381da177e4SLinus Torvalds /* OGF_LINK_POLICY */ 1397c631a67SMarcel Holtmann { 0x00005200, 0x00000000, 0x00000000, 0x00 }, 1401da177e4SLinus Torvalds /* OGF_HOST_CTL */ 1417c631a67SMarcel Holtmann { 0xaab00200, 0x2b402aaa, 0x05220154, 0x00 }, 1421da177e4SLinus Torvalds /* OGF_INFO_PARAM */ 1437c631a67SMarcel Holtmann { 0x000002be, 0x00000000, 0x00000000, 0x00 }, 1441da177e4SLinus Torvalds /* OGF_STATUS_PARAM */ 1457c631a67SMarcel Holtmann { 0x000000ea, 0x00000000, 0x00000000, 0x00 } 1461da177e4SLinus Torvalds } 1471da177e4SLinus Torvalds }; 1481da177e4SLinus Torvalds 1491da177e4SLinus Torvalds static struct bt_sock_list hci_sk_list = { 150d5fb2962SRobert P. J. Day .lock = __RW_LOCK_UNLOCKED(hci_sk_list.lock) 1511da177e4SLinus Torvalds }; 1521da177e4SLinus Torvalds 153f81fe64fSMarcel Holtmann static bool is_filtered_packet(struct sock *sk, struct sk_buff *skb) 154f81fe64fSMarcel Holtmann { 155f81fe64fSMarcel Holtmann struct hci_filter *flt; 156f81fe64fSMarcel Holtmann int flt_type, flt_event; 157f81fe64fSMarcel Holtmann 158f81fe64fSMarcel Holtmann /* Apply filter */ 159f81fe64fSMarcel Holtmann flt = &hci_pi(sk)->filter; 160f81fe64fSMarcel Holtmann 161d79f34e3SMarcel Holtmann flt_type = hci_skb_pkt_type(skb) & HCI_FLT_TYPE_BITS; 162f81fe64fSMarcel Holtmann 163f81fe64fSMarcel Holtmann if (!test_bit(flt_type, &flt->type_mask)) 164f81fe64fSMarcel Holtmann return true; 165f81fe64fSMarcel Holtmann 166f81fe64fSMarcel Holtmann /* Extra filter for event packets only */ 167d79f34e3SMarcel Holtmann if (hci_skb_pkt_type(skb) != HCI_EVENT_PKT) 168f81fe64fSMarcel Holtmann return false; 169f81fe64fSMarcel Holtmann 170f81fe64fSMarcel Holtmann flt_event = (*(__u8 *)skb->data & HCI_FLT_EVENT_BITS); 171f81fe64fSMarcel Holtmann 172f81fe64fSMarcel Holtmann if (!hci_test_bit(flt_event, &flt->event_mask)) 173f81fe64fSMarcel Holtmann return true; 174f81fe64fSMarcel Holtmann 175f81fe64fSMarcel Holtmann /* Check filter only when opcode is set */ 176f81fe64fSMarcel Holtmann if (!flt->opcode) 177f81fe64fSMarcel Holtmann return false; 178f81fe64fSMarcel Holtmann 179f81fe64fSMarcel Holtmann if (flt_event == HCI_EV_CMD_COMPLETE && 180f81fe64fSMarcel Holtmann flt->opcode != get_unaligned((__le16 *)(skb->data + 3))) 181f81fe64fSMarcel Holtmann return true; 182f81fe64fSMarcel Holtmann 183f81fe64fSMarcel Holtmann if (flt_event == HCI_EV_CMD_STATUS && 184f81fe64fSMarcel Holtmann flt->opcode != get_unaligned((__le16 *)(skb->data + 4))) 185f81fe64fSMarcel Holtmann return true; 186f81fe64fSMarcel Holtmann 187f81fe64fSMarcel Holtmann return false; 188f81fe64fSMarcel Holtmann } 189f81fe64fSMarcel Holtmann 1901da177e4SLinus Torvalds /* Send frame to RAW socket */ 191470fe1b5SMarcel Holtmann void hci_send_to_sock(struct hci_dev *hdev, struct sk_buff *skb) 1921da177e4SLinus Torvalds { 1931da177e4SLinus Torvalds struct sock *sk; 194e0edf373SMarcel Holtmann struct sk_buff *skb_copy = NULL; 1951da177e4SLinus Torvalds 1961da177e4SLinus Torvalds BT_DBG("hdev %p len %d", hdev, skb->len); 1971da177e4SLinus Torvalds 1981da177e4SLinus Torvalds read_lock(&hci_sk_list.lock); 199470fe1b5SMarcel Holtmann 200b67bfe0dSSasha Levin sk_for_each(sk, &hci_sk_list.head) { 2011da177e4SLinus Torvalds struct sk_buff *nskb; 2021da177e4SLinus Torvalds 2031da177e4SLinus Torvalds if (sk->sk_state != BT_BOUND || hci_pi(sk)->hdev != hdev) 2041da177e4SLinus Torvalds continue; 2051da177e4SLinus Torvalds 2061da177e4SLinus Torvalds /* Don't send frame to the socket it came from */ 2071da177e4SLinus Torvalds if (skb->sk == sk) 2081da177e4SLinus Torvalds continue; 2091da177e4SLinus Torvalds 21023500189SMarcel Holtmann if (hci_pi(sk)->channel == HCI_CHANNEL_RAW) { 211d79f34e3SMarcel Holtmann if (hci_skb_pkt_type(skb) != HCI_COMMAND_PKT && 212d79f34e3SMarcel Holtmann hci_skb_pkt_type(skb) != HCI_EVENT_PKT && 213d79f34e3SMarcel Holtmann hci_skb_pkt_type(skb) != HCI_ACLDATA_PKT && 214d79f34e3SMarcel Holtmann hci_skb_pkt_type(skb) != HCI_SCODATA_PKT) 215bb77543eSMarcel Holtmann continue; 216f81fe64fSMarcel Holtmann if (is_filtered_packet(sk, skb)) 2171da177e4SLinus Torvalds continue; 21823500189SMarcel Holtmann } else if (hci_pi(sk)->channel == HCI_CHANNEL_USER) { 21923500189SMarcel Holtmann if (!bt_cb(skb)->incoming) 22023500189SMarcel Holtmann continue; 221d79f34e3SMarcel Holtmann if (hci_skb_pkt_type(skb) != HCI_EVENT_PKT && 222d79f34e3SMarcel Holtmann hci_skb_pkt_type(skb) != HCI_ACLDATA_PKT && 223d79f34e3SMarcel Holtmann hci_skb_pkt_type(skb) != HCI_SCODATA_PKT) 22423500189SMarcel Holtmann continue; 22523500189SMarcel Holtmann } else { 22623500189SMarcel Holtmann /* Don't send frame to other channel types */ 22723500189SMarcel Holtmann continue; 22823500189SMarcel Holtmann } 2291da177e4SLinus Torvalds 230e0edf373SMarcel Holtmann if (!skb_copy) { 231e0edf373SMarcel Holtmann /* Create a private copy with headroom */ 232bad93e9dSOctavian Purdila skb_copy = __pskb_copy_fclone(skb, 1, GFP_ATOMIC, true); 233e0edf373SMarcel Holtmann if (!skb_copy) 2341da177e4SLinus Torvalds continue; 2351da177e4SLinus Torvalds 2361da177e4SLinus Torvalds /* Put type byte before the data */ 237d79f34e3SMarcel Holtmann memcpy(skb_push(skb_copy, 1), &hci_skb_pkt_type(skb), 1); 238e0edf373SMarcel Holtmann } 239e0edf373SMarcel Holtmann 240e0edf373SMarcel Holtmann nskb = skb_clone(skb_copy, GFP_ATOMIC); 241e0edf373SMarcel Holtmann if (!nskb) 242e0edf373SMarcel Holtmann continue; 2431da177e4SLinus Torvalds 2441da177e4SLinus Torvalds if (sock_queue_rcv_skb(sk, nskb)) 2451da177e4SLinus Torvalds kfree_skb(nskb); 2461da177e4SLinus Torvalds } 247470fe1b5SMarcel Holtmann 248470fe1b5SMarcel Holtmann read_unlock(&hci_sk_list.lock); 249e0edf373SMarcel Holtmann 250e0edf373SMarcel Holtmann kfree_skb(skb_copy); 251470fe1b5SMarcel Holtmann } 252470fe1b5SMarcel Holtmann 2537129069eSJohan Hedberg /* Send frame to sockets with specific channel */ 2547129069eSJohan Hedberg void hci_send_to_channel(unsigned short channel, struct sk_buff *skb, 255c08b1a1dSMarcel Holtmann int flag, struct sock *skip_sk) 256470fe1b5SMarcel Holtmann { 257470fe1b5SMarcel Holtmann struct sock *sk; 258470fe1b5SMarcel Holtmann 2597129069eSJohan Hedberg BT_DBG("channel %u len %d", channel, skb->len); 260470fe1b5SMarcel Holtmann 261470fe1b5SMarcel Holtmann read_lock(&hci_sk_list.lock); 262470fe1b5SMarcel Holtmann 263b67bfe0dSSasha Levin sk_for_each(sk, &hci_sk_list.head) { 264470fe1b5SMarcel Holtmann struct sk_buff *nskb; 265470fe1b5SMarcel Holtmann 266c08b1a1dSMarcel Holtmann /* Ignore socket without the flag set */ 267c85be545SMarcel Holtmann if (!hci_sock_test_flag(sk, flag)) 268c08b1a1dSMarcel Holtmann continue; 269c08b1a1dSMarcel Holtmann 270470fe1b5SMarcel Holtmann /* Skip the original socket */ 271470fe1b5SMarcel Holtmann if (sk == skip_sk) 272470fe1b5SMarcel Holtmann continue; 273470fe1b5SMarcel Holtmann 274470fe1b5SMarcel Holtmann if (sk->sk_state != BT_BOUND) 275470fe1b5SMarcel Holtmann continue; 276470fe1b5SMarcel Holtmann 2777129069eSJohan Hedberg if (hci_pi(sk)->channel != channel) 278d7f72f61SMarcel Holtmann continue; 279d7f72f61SMarcel Holtmann 280d7f72f61SMarcel Holtmann nskb = skb_clone(skb, GFP_ATOMIC); 281d7f72f61SMarcel Holtmann if (!nskb) 282d7f72f61SMarcel Holtmann continue; 283d7f72f61SMarcel Holtmann 284d7f72f61SMarcel Holtmann if (sock_queue_rcv_skb(sk, nskb)) 285d7f72f61SMarcel Holtmann kfree_skb(nskb); 286d7f72f61SMarcel Holtmann } 287d7f72f61SMarcel Holtmann 288d7f72f61SMarcel Holtmann read_unlock(&hci_sk_list.lock); 289d7f72f61SMarcel Holtmann } 290d7f72f61SMarcel Holtmann 291cd82e61cSMarcel Holtmann /* Send frame to monitor socket */ 292cd82e61cSMarcel Holtmann void hci_send_to_monitor(struct hci_dev *hdev, struct sk_buff *skb) 293cd82e61cSMarcel Holtmann { 294cd82e61cSMarcel Holtmann struct sk_buff *skb_copy = NULL; 2952b531294SMarcel Holtmann struct hci_mon_hdr *hdr; 296cd82e61cSMarcel Holtmann __le16 opcode; 297cd82e61cSMarcel Holtmann 298cd82e61cSMarcel Holtmann if (!atomic_read(&monitor_promisc)) 299cd82e61cSMarcel Holtmann return; 300cd82e61cSMarcel Holtmann 301cd82e61cSMarcel Holtmann BT_DBG("hdev %p len %d", hdev, skb->len); 302cd82e61cSMarcel Holtmann 303d79f34e3SMarcel Holtmann switch (hci_skb_pkt_type(skb)) { 304cd82e61cSMarcel Holtmann case HCI_COMMAND_PKT: 305dcf4adbfSJoe Perches opcode = cpu_to_le16(HCI_MON_COMMAND_PKT); 306cd82e61cSMarcel Holtmann break; 307cd82e61cSMarcel Holtmann case HCI_EVENT_PKT: 308dcf4adbfSJoe Perches opcode = cpu_to_le16(HCI_MON_EVENT_PKT); 309cd82e61cSMarcel Holtmann break; 310cd82e61cSMarcel Holtmann case HCI_ACLDATA_PKT: 311cd82e61cSMarcel Holtmann if (bt_cb(skb)->incoming) 312dcf4adbfSJoe Perches opcode = cpu_to_le16(HCI_MON_ACL_RX_PKT); 313cd82e61cSMarcel Holtmann else 314dcf4adbfSJoe Perches opcode = cpu_to_le16(HCI_MON_ACL_TX_PKT); 315cd82e61cSMarcel Holtmann break; 316cd82e61cSMarcel Holtmann case HCI_SCODATA_PKT: 317cd82e61cSMarcel Holtmann if (bt_cb(skb)->incoming) 318dcf4adbfSJoe Perches opcode = cpu_to_le16(HCI_MON_SCO_RX_PKT); 319cd82e61cSMarcel Holtmann else 320dcf4adbfSJoe Perches opcode = cpu_to_le16(HCI_MON_SCO_TX_PKT); 321cd82e61cSMarcel Holtmann break; 322e875ff84SMarcel Holtmann case HCI_DIAG_PKT: 323e875ff84SMarcel Holtmann opcode = cpu_to_le16(HCI_MON_VENDOR_DIAG); 324e875ff84SMarcel Holtmann break; 325cd82e61cSMarcel Holtmann default: 326cd82e61cSMarcel Holtmann return; 327cd82e61cSMarcel Holtmann } 328cd82e61cSMarcel Holtmann 3292b531294SMarcel Holtmann /* Create a private copy with headroom */ 3302b531294SMarcel Holtmann skb_copy = __pskb_copy_fclone(skb, HCI_MON_HDR_SIZE, GFP_ATOMIC, true); 3312b531294SMarcel Holtmann if (!skb_copy) 3322b531294SMarcel Holtmann return; 3332b531294SMarcel Holtmann 3342b531294SMarcel Holtmann /* Put header before the data */ 3352b531294SMarcel Holtmann hdr = (void *)skb_push(skb_copy, HCI_MON_HDR_SIZE); 3362b531294SMarcel Holtmann hdr->opcode = opcode; 3372b531294SMarcel Holtmann hdr->index = cpu_to_le16(hdev->id); 3382b531294SMarcel Holtmann hdr->len = cpu_to_le16(skb->len); 3392b531294SMarcel Holtmann 340c08b1a1dSMarcel Holtmann hci_send_to_channel(HCI_CHANNEL_MONITOR, skb_copy, 341c08b1a1dSMarcel Holtmann HCI_SOCK_TRUSTED, NULL); 342cd82e61cSMarcel Holtmann kfree_skb(skb_copy); 343cd82e61cSMarcel Holtmann } 344cd82e61cSMarcel Holtmann 34538ceaa00SMarcel Holtmann void hci_send_monitor_ctrl_event(struct hci_dev *hdev, u16 event, 34638ceaa00SMarcel Holtmann void *data, u16 data_len, ktime_t tstamp, 34738ceaa00SMarcel Holtmann int flag, struct sock *skip_sk) 34838ceaa00SMarcel Holtmann { 34938ceaa00SMarcel Holtmann struct sock *sk; 35038ceaa00SMarcel Holtmann __le16 index; 35138ceaa00SMarcel Holtmann 35238ceaa00SMarcel Holtmann if (hdev) 35338ceaa00SMarcel Holtmann index = cpu_to_le16(hdev->id); 35438ceaa00SMarcel Holtmann else 35538ceaa00SMarcel Holtmann index = cpu_to_le16(MGMT_INDEX_NONE); 35638ceaa00SMarcel Holtmann 35738ceaa00SMarcel Holtmann read_lock(&hci_sk_list.lock); 35838ceaa00SMarcel Holtmann 35938ceaa00SMarcel Holtmann sk_for_each(sk, &hci_sk_list.head) { 36038ceaa00SMarcel Holtmann struct hci_mon_hdr *hdr; 36138ceaa00SMarcel Holtmann struct sk_buff *skb; 36238ceaa00SMarcel Holtmann 36338ceaa00SMarcel Holtmann if (hci_pi(sk)->channel != HCI_CHANNEL_CONTROL) 36438ceaa00SMarcel Holtmann continue; 36538ceaa00SMarcel Holtmann 36638ceaa00SMarcel Holtmann /* Ignore socket without the flag set */ 36738ceaa00SMarcel Holtmann if (!hci_sock_test_flag(sk, flag)) 36838ceaa00SMarcel Holtmann continue; 36938ceaa00SMarcel Holtmann 37038ceaa00SMarcel Holtmann /* Skip the original socket */ 37138ceaa00SMarcel Holtmann if (sk == skip_sk) 37238ceaa00SMarcel Holtmann continue; 37338ceaa00SMarcel Holtmann 37438ceaa00SMarcel Holtmann skb = bt_skb_alloc(6 + data_len, GFP_ATOMIC); 37538ceaa00SMarcel Holtmann if (!skb) 37638ceaa00SMarcel Holtmann continue; 37738ceaa00SMarcel Holtmann 37838ceaa00SMarcel Holtmann put_unaligned_le32(hci_pi(sk)->cookie, skb_put(skb, 4)); 37938ceaa00SMarcel Holtmann put_unaligned_le16(event, skb_put(skb, 2)); 38038ceaa00SMarcel Holtmann 38138ceaa00SMarcel Holtmann if (data) 38238ceaa00SMarcel Holtmann memcpy(skb_put(skb, data_len), data, data_len); 38338ceaa00SMarcel Holtmann 38438ceaa00SMarcel Holtmann skb->tstamp = tstamp; 38538ceaa00SMarcel Holtmann 38638ceaa00SMarcel Holtmann hdr = (void *)skb_push(skb, HCI_MON_HDR_SIZE); 38738ceaa00SMarcel Holtmann hdr->opcode = cpu_to_le16(HCI_MON_CTRL_EVENT); 38838ceaa00SMarcel Holtmann hdr->index = index; 38938ceaa00SMarcel Holtmann hdr->len = cpu_to_le16(skb->len - HCI_MON_HDR_SIZE); 39038ceaa00SMarcel Holtmann 39138ceaa00SMarcel Holtmann hci_send_to_channel(HCI_CHANNEL_MONITOR, skb, 39238ceaa00SMarcel Holtmann HCI_SOCK_TRUSTED, NULL); 39338ceaa00SMarcel Holtmann kfree_skb(skb); 39438ceaa00SMarcel Holtmann } 39538ceaa00SMarcel Holtmann 39638ceaa00SMarcel Holtmann read_unlock(&hci_sk_list.lock); 39738ceaa00SMarcel Holtmann } 39838ceaa00SMarcel Holtmann 399cd82e61cSMarcel Holtmann static struct sk_buff *create_monitor_event(struct hci_dev *hdev, int event) 400cd82e61cSMarcel Holtmann { 401cd82e61cSMarcel Holtmann struct hci_mon_hdr *hdr; 402cd82e61cSMarcel Holtmann struct hci_mon_new_index *ni; 4036c566dd5SMarcel Holtmann struct hci_mon_index_info *ii; 404cd82e61cSMarcel Holtmann struct sk_buff *skb; 405cd82e61cSMarcel Holtmann __le16 opcode; 406cd82e61cSMarcel Holtmann 407cd82e61cSMarcel Holtmann switch (event) { 408cd82e61cSMarcel Holtmann case HCI_DEV_REG: 409cd82e61cSMarcel Holtmann skb = bt_skb_alloc(HCI_MON_NEW_INDEX_SIZE, GFP_ATOMIC); 410cd82e61cSMarcel Holtmann if (!skb) 411cd82e61cSMarcel Holtmann return NULL; 412cd82e61cSMarcel Holtmann 413cd82e61cSMarcel Holtmann ni = (void *)skb_put(skb, HCI_MON_NEW_INDEX_SIZE); 414cd82e61cSMarcel Holtmann ni->type = hdev->dev_type; 415cd82e61cSMarcel Holtmann ni->bus = hdev->bus; 416cd82e61cSMarcel Holtmann bacpy(&ni->bdaddr, &hdev->bdaddr); 417cd82e61cSMarcel Holtmann memcpy(ni->name, hdev->name, 8); 418cd82e61cSMarcel Holtmann 419dcf4adbfSJoe Perches opcode = cpu_to_le16(HCI_MON_NEW_INDEX); 420cd82e61cSMarcel Holtmann break; 421cd82e61cSMarcel Holtmann 422cd82e61cSMarcel Holtmann case HCI_DEV_UNREG: 423cd82e61cSMarcel Holtmann skb = bt_skb_alloc(0, GFP_ATOMIC); 424cd82e61cSMarcel Holtmann if (!skb) 425cd82e61cSMarcel Holtmann return NULL; 426cd82e61cSMarcel Holtmann 427dcf4adbfSJoe Perches opcode = cpu_to_le16(HCI_MON_DEL_INDEX); 428cd82e61cSMarcel Holtmann break; 429cd82e61cSMarcel Holtmann 430e131d74aSMarcel Holtmann case HCI_DEV_SETUP: 431e131d74aSMarcel Holtmann if (hdev->manufacturer == 0xffff) 432e131d74aSMarcel Holtmann return NULL; 433e131d74aSMarcel Holtmann 434e131d74aSMarcel Holtmann /* fall through */ 435e131d74aSMarcel Holtmann 4366c566dd5SMarcel Holtmann case HCI_DEV_UP: 4376c566dd5SMarcel Holtmann skb = bt_skb_alloc(HCI_MON_INDEX_INFO_SIZE, GFP_ATOMIC); 4386c566dd5SMarcel Holtmann if (!skb) 4396c566dd5SMarcel Holtmann return NULL; 4406c566dd5SMarcel Holtmann 4416c566dd5SMarcel Holtmann ii = (void *)skb_put(skb, HCI_MON_INDEX_INFO_SIZE); 4426c566dd5SMarcel Holtmann bacpy(&ii->bdaddr, &hdev->bdaddr); 4436c566dd5SMarcel Holtmann ii->manufacturer = cpu_to_le16(hdev->manufacturer); 4446c566dd5SMarcel Holtmann 4456c566dd5SMarcel Holtmann opcode = cpu_to_le16(HCI_MON_INDEX_INFO); 4466c566dd5SMarcel Holtmann break; 4476c566dd5SMarcel Holtmann 44822db3cbcSMarcel Holtmann case HCI_DEV_OPEN: 44922db3cbcSMarcel Holtmann skb = bt_skb_alloc(0, GFP_ATOMIC); 45022db3cbcSMarcel Holtmann if (!skb) 45122db3cbcSMarcel Holtmann return NULL; 45222db3cbcSMarcel Holtmann 45322db3cbcSMarcel Holtmann opcode = cpu_to_le16(HCI_MON_OPEN_INDEX); 45422db3cbcSMarcel Holtmann break; 45522db3cbcSMarcel Holtmann 45622db3cbcSMarcel Holtmann case HCI_DEV_CLOSE: 45722db3cbcSMarcel Holtmann skb = bt_skb_alloc(0, GFP_ATOMIC); 45822db3cbcSMarcel Holtmann if (!skb) 45922db3cbcSMarcel Holtmann return NULL; 46022db3cbcSMarcel Holtmann 46122db3cbcSMarcel Holtmann opcode = cpu_to_le16(HCI_MON_CLOSE_INDEX); 46222db3cbcSMarcel Holtmann break; 46322db3cbcSMarcel Holtmann 464cd82e61cSMarcel Holtmann default: 465cd82e61cSMarcel Holtmann return NULL; 466cd82e61cSMarcel Holtmann } 467cd82e61cSMarcel Holtmann 468cd82e61cSMarcel Holtmann __net_timestamp(skb); 469cd82e61cSMarcel Holtmann 470cd82e61cSMarcel Holtmann hdr = (void *)skb_push(skb, HCI_MON_HDR_SIZE); 471cd82e61cSMarcel Holtmann hdr->opcode = opcode; 472cd82e61cSMarcel Holtmann hdr->index = cpu_to_le16(hdev->id); 473cd82e61cSMarcel Holtmann hdr->len = cpu_to_le16(skb->len - HCI_MON_HDR_SIZE); 474cd82e61cSMarcel Holtmann 475cd82e61cSMarcel Holtmann return skb; 476cd82e61cSMarcel Holtmann } 477cd82e61cSMarcel Holtmann 478249fa169SMarcel Holtmann static struct sk_buff *create_monitor_ctrl_open(struct sock *sk) 479249fa169SMarcel Holtmann { 480249fa169SMarcel Holtmann struct hci_mon_hdr *hdr; 481249fa169SMarcel Holtmann struct sk_buff *skb; 482249fa169SMarcel Holtmann u16 format = 0x0002; 483249fa169SMarcel Holtmann u8 ver[3]; 484249fa169SMarcel Holtmann u32 flags; 485249fa169SMarcel Holtmann 486249fa169SMarcel Holtmann skb = bt_skb_alloc(14 + TASK_COMM_LEN , GFP_ATOMIC); 487249fa169SMarcel Holtmann if (!skb) 488249fa169SMarcel Holtmann return NULL; 489249fa169SMarcel Holtmann 490249fa169SMarcel Holtmann mgmt_fill_version_info(ver); 491249fa169SMarcel Holtmann flags = hci_sock_test_flag(sk, HCI_SOCK_TRUSTED) ? 0x1 : 0x0; 492249fa169SMarcel Holtmann 493249fa169SMarcel Holtmann put_unaligned_le32(hci_pi(sk)->cookie, skb_put(skb, 4)); 494249fa169SMarcel Holtmann put_unaligned_le16(format, skb_put(skb, 2)); 495249fa169SMarcel Holtmann memcpy(skb_put(skb, sizeof(ver)), ver, sizeof(ver)); 496249fa169SMarcel Holtmann put_unaligned_le32(flags, skb_put(skb, 4)); 497249fa169SMarcel Holtmann *skb_put(skb, 1) = TASK_COMM_LEN; 498249fa169SMarcel Holtmann memcpy(skb_put(skb, TASK_COMM_LEN), hci_pi(sk)->comm, TASK_COMM_LEN); 499249fa169SMarcel Holtmann 500249fa169SMarcel Holtmann __net_timestamp(skb); 501249fa169SMarcel Holtmann 502249fa169SMarcel Holtmann hdr = (void *)skb_push(skb, HCI_MON_HDR_SIZE); 503249fa169SMarcel Holtmann hdr->opcode = cpu_to_le16(HCI_MON_CTRL_OPEN); 504249fa169SMarcel Holtmann hdr->index = cpu_to_le16(HCI_DEV_NONE); 505249fa169SMarcel Holtmann hdr->len = cpu_to_le16(skb->len - HCI_MON_HDR_SIZE); 506249fa169SMarcel Holtmann 507249fa169SMarcel Holtmann return skb; 508249fa169SMarcel Holtmann } 509249fa169SMarcel Holtmann 510249fa169SMarcel Holtmann static struct sk_buff *create_monitor_ctrl_close(struct sock *sk) 511249fa169SMarcel Holtmann { 512249fa169SMarcel Holtmann struct hci_mon_hdr *hdr; 513249fa169SMarcel Holtmann struct sk_buff *skb; 514249fa169SMarcel Holtmann 515249fa169SMarcel Holtmann skb = bt_skb_alloc(4, GFP_ATOMIC); 516249fa169SMarcel Holtmann if (!skb) 517249fa169SMarcel Holtmann return NULL; 518249fa169SMarcel Holtmann 519249fa169SMarcel Holtmann put_unaligned_le32(hci_pi(sk)->cookie, skb_put(skb, 4)); 520249fa169SMarcel Holtmann 521249fa169SMarcel Holtmann __net_timestamp(skb); 522249fa169SMarcel Holtmann 523249fa169SMarcel Holtmann hdr = (void *)skb_push(skb, HCI_MON_HDR_SIZE); 524249fa169SMarcel Holtmann hdr->opcode = cpu_to_le16(HCI_MON_CTRL_CLOSE); 525249fa169SMarcel Holtmann hdr->index = cpu_to_le16(HCI_DEV_NONE); 526249fa169SMarcel Holtmann hdr->len = cpu_to_le16(skb->len - HCI_MON_HDR_SIZE); 527249fa169SMarcel Holtmann 528249fa169SMarcel Holtmann return skb; 529249fa169SMarcel Holtmann } 530249fa169SMarcel Holtmann 53138ceaa00SMarcel Holtmann static struct sk_buff *create_monitor_ctrl_command(struct sock *sk, u16 index, 53238ceaa00SMarcel Holtmann u16 opcode, u16 len, 53338ceaa00SMarcel Holtmann const void *buf) 53438ceaa00SMarcel Holtmann { 53538ceaa00SMarcel Holtmann struct hci_mon_hdr *hdr; 53638ceaa00SMarcel Holtmann struct sk_buff *skb; 53738ceaa00SMarcel Holtmann 53838ceaa00SMarcel Holtmann skb = bt_skb_alloc(6 + len, GFP_ATOMIC); 53938ceaa00SMarcel Holtmann if (!skb) 54038ceaa00SMarcel Holtmann return NULL; 54138ceaa00SMarcel Holtmann 54238ceaa00SMarcel Holtmann put_unaligned_le32(hci_pi(sk)->cookie, skb_put(skb, 4)); 54338ceaa00SMarcel Holtmann put_unaligned_le16(opcode, skb_put(skb, 2)); 54438ceaa00SMarcel Holtmann 54538ceaa00SMarcel Holtmann if (buf) 54638ceaa00SMarcel Holtmann memcpy(skb_put(skb, len), buf, len); 54738ceaa00SMarcel Holtmann 54838ceaa00SMarcel Holtmann __net_timestamp(skb); 54938ceaa00SMarcel Holtmann 55038ceaa00SMarcel Holtmann hdr = (void *)skb_push(skb, HCI_MON_HDR_SIZE); 55138ceaa00SMarcel Holtmann hdr->opcode = cpu_to_le16(HCI_MON_CTRL_COMMAND); 55238ceaa00SMarcel Holtmann hdr->index = cpu_to_le16(index); 55338ceaa00SMarcel Holtmann hdr->len = cpu_to_le16(skb->len - HCI_MON_HDR_SIZE); 55438ceaa00SMarcel Holtmann 55538ceaa00SMarcel Holtmann return skb; 55638ceaa00SMarcel Holtmann } 55738ceaa00SMarcel Holtmann 558787b306cSJohannes Berg static void __printf(2, 3) 559787b306cSJohannes Berg send_monitor_note(struct sock *sk, const char *fmt, ...) 560dd31506dSMarcel Holtmann { 561787b306cSJohannes Berg size_t len; 562dd31506dSMarcel Holtmann struct hci_mon_hdr *hdr; 563dd31506dSMarcel Holtmann struct sk_buff *skb; 564787b306cSJohannes Berg va_list args; 565787b306cSJohannes Berg 566787b306cSJohannes Berg va_start(args, fmt); 567787b306cSJohannes Berg len = vsnprintf(NULL, 0, fmt, args); 568787b306cSJohannes Berg va_end(args); 569dd31506dSMarcel Holtmann 570dd31506dSMarcel Holtmann skb = bt_skb_alloc(len + 1, GFP_ATOMIC); 571dd31506dSMarcel Holtmann if (!skb) 572dd31506dSMarcel Holtmann return; 573dd31506dSMarcel Holtmann 574787b306cSJohannes Berg va_start(args, fmt); 575787b306cSJohannes Berg vsprintf(skb_put(skb, len), fmt, args); 576787b306cSJohannes Berg *skb_put(skb, 1) = 0; 577787b306cSJohannes Berg va_end(args); 578dd31506dSMarcel Holtmann 579dd31506dSMarcel Holtmann __net_timestamp(skb); 580dd31506dSMarcel Holtmann 581dd31506dSMarcel Holtmann hdr = (void *)skb_push(skb, HCI_MON_HDR_SIZE); 582dd31506dSMarcel Holtmann hdr->opcode = cpu_to_le16(HCI_MON_SYSTEM_NOTE); 583dd31506dSMarcel Holtmann hdr->index = cpu_to_le16(HCI_DEV_NONE); 584dd31506dSMarcel Holtmann hdr->len = cpu_to_le16(skb->len - HCI_MON_HDR_SIZE); 585dd31506dSMarcel Holtmann 586dd31506dSMarcel Holtmann if (sock_queue_rcv_skb(sk, skb)) 587dd31506dSMarcel Holtmann kfree_skb(skb); 588dd31506dSMarcel Holtmann } 589dd31506dSMarcel Holtmann 590cd82e61cSMarcel Holtmann static void send_monitor_replay(struct sock *sk) 591cd82e61cSMarcel Holtmann { 592cd82e61cSMarcel Holtmann struct hci_dev *hdev; 593cd82e61cSMarcel Holtmann 594cd82e61cSMarcel Holtmann read_lock(&hci_dev_list_lock); 595cd82e61cSMarcel Holtmann 596cd82e61cSMarcel Holtmann list_for_each_entry(hdev, &hci_dev_list, list) { 597cd82e61cSMarcel Holtmann struct sk_buff *skb; 598cd82e61cSMarcel Holtmann 599cd82e61cSMarcel Holtmann skb = create_monitor_event(hdev, HCI_DEV_REG); 600cd82e61cSMarcel Holtmann if (!skb) 601cd82e61cSMarcel Holtmann continue; 602cd82e61cSMarcel Holtmann 603cd82e61cSMarcel Holtmann if (sock_queue_rcv_skb(sk, skb)) 604cd82e61cSMarcel Holtmann kfree_skb(skb); 60522db3cbcSMarcel Holtmann 60622db3cbcSMarcel Holtmann if (!test_bit(HCI_RUNNING, &hdev->flags)) 60722db3cbcSMarcel Holtmann continue; 60822db3cbcSMarcel Holtmann 60922db3cbcSMarcel Holtmann skb = create_monitor_event(hdev, HCI_DEV_OPEN); 61022db3cbcSMarcel Holtmann if (!skb) 61122db3cbcSMarcel Holtmann continue; 61222db3cbcSMarcel Holtmann 61322db3cbcSMarcel Holtmann if (sock_queue_rcv_skb(sk, skb)) 61422db3cbcSMarcel Holtmann kfree_skb(skb); 6156c566dd5SMarcel Holtmann 616e131d74aSMarcel Holtmann if (test_bit(HCI_UP, &hdev->flags)) 6176c566dd5SMarcel Holtmann skb = create_monitor_event(hdev, HCI_DEV_UP); 618e131d74aSMarcel Holtmann else if (hci_dev_test_flag(hdev, HCI_SETUP)) 619e131d74aSMarcel Holtmann skb = create_monitor_event(hdev, HCI_DEV_SETUP); 620e131d74aSMarcel Holtmann else 621e131d74aSMarcel Holtmann skb = NULL; 6226c566dd5SMarcel Holtmann 623e131d74aSMarcel Holtmann if (skb) { 6246c566dd5SMarcel Holtmann if (sock_queue_rcv_skb(sk, skb)) 6256c566dd5SMarcel Holtmann kfree_skb(skb); 626cd82e61cSMarcel Holtmann } 627e131d74aSMarcel Holtmann } 628cd82e61cSMarcel Holtmann 629cd82e61cSMarcel Holtmann read_unlock(&hci_dev_list_lock); 630cd82e61cSMarcel Holtmann } 631cd82e61cSMarcel Holtmann 632249fa169SMarcel Holtmann static void send_monitor_control_replay(struct sock *mon_sk) 633249fa169SMarcel Holtmann { 634249fa169SMarcel Holtmann struct sock *sk; 635249fa169SMarcel Holtmann 636249fa169SMarcel Holtmann read_lock(&hci_sk_list.lock); 637249fa169SMarcel Holtmann 638249fa169SMarcel Holtmann sk_for_each(sk, &hci_sk_list.head) { 639249fa169SMarcel Holtmann struct sk_buff *skb; 640249fa169SMarcel Holtmann 641249fa169SMarcel Holtmann if (hci_pi(sk)->channel != HCI_CHANNEL_CONTROL) 642249fa169SMarcel Holtmann continue; 643249fa169SMarcel Holtmann 644249fa169SMarcel Holtmann skb = create_monitor_ctrl_open(sk); 645249fa169SMarcel Holtmann if (!skb) 646249fa169SMarcel Holtmann continue; 647249fa169SMarcel Holtmann 648249fa169SMarcel Holtmann if (sock_queue_rcv_skb(mon_sk, skb)) 649249fa169SMarcel Holtmann kfree_skb(skb); 650249fa169SMarcel Holtmann } 651249fa169SMarcel Holtmann 652249fa169SMarcel Holtmann read_unlock(&hci_sk_list.lock); 653249fa169SMarcel Holtmann } 654249fa169SMarcel Holtmann 655040030efSMarcel Holtmann /* Generate internal stack event */ 656040030efSMarcel Holtmann static void hci_si_event(struct hci_dev *hdev, int type, int dlen, void *data) 657040030efSMarcel Holtmann { 658040030efSMarcel Holtmann struct hci_event_hdr *hdr; 659040030efSMarcel Holtmann struct hci_ev_stack_internal *ev; 660040030efSMarcel Holtmann struct sk_buff *skb; 661040030efSMarcel Holtmann 662040030efSMarcel Holtmann skb = bt_skb_alloc(HCI_EVENT_HDR_SIZE + sizeof(*ev) + dlen, GFP_ATOMIC); 663040030efSMarcel Holtmann if (!skb) 664040030efSMarcel Holtmann return; 665040030efSMarcel Holtmann 666040030efSMarcel Holtmann hdr = (void *)skb_put(skb, HCI_EVENT_HDR_SIZE); 667040030efSMarcel Holtmann hdr->evt = HCI_EV_STACK_INTERNAL; 668040030efSMarcel Holtmann hdr->plen = sizeof(*ev) + dlen; 669040030efSMarcel Holtmann 670040030efSMarcel Holtmann ev = (void *)skb_put(skb, sizeof(*ev) + dlen); 671040030efSMarcel Holtmann ev->type = type; 672040030efSMarcel Holtmann memcpy(ev->data, data, dlen); 673040030efSMarcel Holtmann 674040030efSMarcel Holtmann bt_cb(skb)->incoming = 1; 675040030efSMarcel Holtmann __net_timestamp(skb); 676040030efSMarcel Holtmann 677d79f34e3SMarcel Holtmann hci_skb_pkt_type(skb) = HCI_EVENT_PKT; 678040030efSMarcel Holtmann hci_send_to_sock(hdev, skb); 679040030efSMarcel Holtmann kfree_skb(skb); 680040030efSMarcel Holtmann } 681040030efSMarcel Holtmann 682040030efSMarcel Holtmann void hci_sock_dev_event(struct hci_dev *hdev, int event) 683040030efSMarcel Holtmann { 684040030efSMarcel Holtmann BT_DBG("hdev %s event %d", hdev->name, event); 685040030efSMarcel Holtmann 686cd82e61cSMarcel Holtmann if (atomic_read(&monitor_promisc)) { 687cd82e61cSMarcel Holtmann struct sk_buff *skb; 688cd82e61cSMarcel Holtmann 689ed1b28a4SMarcel Holtmann /* Send event to monitor */ 690cd82e61cSMarcel Holtmann skb = create_monitor_event(hdev, event); 691cd82e61cSMarcel Holtmann if (skb) { 692c08b1a1dSMarcel Holtmann hci_send_to_channel(HCI_CHANNEL_MONITOR, skb, 693c08b1a1dSMarcel Holtmann HCI_SOCK_TRUSTED, NULL); 694cd82e61cSMarcel Holtmann kfree_skb(skb); 695cd82e61cSMarcel Holtmann } 696cd82e61cSMarcel Holtmann } 697cd82e61cSMarcel Holtmann 698ed1b28a4SMarcel Holtmann if (event <= HCI_DEV_DOWN) { 699ed1b28a4SMarcel Holtmann struct hci_ev_si_device ev; 700ed1b28a4SMarcel Holtmann 701040030efSMarcel Holtmann /* Send event to sockets */ 702040030efSMarcel Holtmann ev.event = event; 703040030efSMarcel Holtmann ev.dev_id = hdev->id; 704040030efSMarcel Holtmann hci_si_event(NULL, HCI_EV_SI_DEVICE, sizeof(ev), &ev); 705ed1b28a4SMarcel Holtmann } 706040030efSMarcel Holtmann 707040030efSMarcel Holtmann if (event == HCI_DEV_UNREG) { 708040030efSMarcel Holtmann struct sock *sk; 709040030efSMarcel Holtmann 710040030efSMarcel Holtmann /* Detach sockets from device */ 711040030efSMarcel Holtmann read_lock(&hci_sk_list.lock); 712b67bfe0dSSasha Levin sk_for_each(sk, &hci_sk_list.head) { 713040030efSMarcel Holtmann bh_lock_sock_nested(sk); 714040030efSMarcel Holtmann if (hci_pi(sk)->hdev == hdev) { 715040030efSMarcel Holtmann hci_pi(sk)->hdev = NULL; 716040030efSMarcel Holtmann sk->sk_err = EPIPE; 717040030efSMarcel Holtmann sk->sk_state = BT_OPEN; 718040030efSMarcel Holtmann sk->sk_state_change(sk); 719040030efSMarcel Holtmann 720040030efSMarcel Holtmann hci_dev_put(hdev); 721040030efSMarcel Holtmann } 722040030efSMarcel Holtmann bh_unlock_sock(sk); 723040030efSMarcel Holtmann } 724040030efSMarcel Holtmann read_unlock(&hci_sk_list.lock); 725040030efSMarcel Holtmann } 726040030efSMarcel Holtmann } 727040030efSMarcel Holtmann 728801c1e8dSJohan Hedberg static struct hci_mgmt_chan *__hci_mgmt_chan_find(unsigned short channel) 729801c1e8dSJohan Hedberg { 730801c1e8dSJohan Hedberg struct hci_mgmt_chan *c; 731801c1e8dSJohan Hedberg 732801c1e8dSJohan Hedberg list_for_each_entry(c, &mgmt_chan_list, list) { 733801c1e8dSJohan Hedberg if (c->channel == channel) 734801c1e8dSJohan Hedberg return c; 735801c1e8dSJohan Hedberg } 736801c1e8dSJohan Hedberg 737801c1e8dSJohan Hedberg return NULL; 738801c1e8dSJohan Hedberg } 739801c1e8dSJohan Hedberg 740801c1e8dSJohan Hedberg static struct hci_mgmt_chan *hci_mgmt_chan_find(unsigned short channel) 741801c1e8dSJohan Hedberg { 742801c1e8dSJohan Hedberg struct hci_mgmt_chan *c; 743801c1e8dSJohan Hedberg 744801c1e8dSJohan Hedberg mutex_lock(&mgmt_chan_list_lock); 745801c1e8dSJohan Hedberg c = __hci_mgmt_chan_find(channel); 746801c1e8dSJohan Hedberg mutex_unlock(&mgmt_chan_list_lock); 747801c1e8dSJohan Hedberg 748801c1e8dSJohan Hedberg return c; 749801c1e8dSJohan Hedberg } 750801c1e8dSJohan Hedberg 751801c1e8dSJohan Hedberg int hci_mgmt_chan_register(struct hci_mgmt_chan *c) 752801c1e8dSJohan Hedberg { 753801c1e8dSJohan Hedberg if (c->channel < HCI_CHANNEL_CONTROL) 754801c1e8dSJohan Hedberg return -EINVAL; 755801c1e8dSJohan Hedberg 756801c1e8dSJohan Hedberg mutex_lock(&mgmt_chan_list_lock); 757801c1e8dSJohan Hedberg if (__hci_mgmt_chan_find(c->channel)) { 758801c1e8dSJohan Hedberg mutex_unlock(&mgmt_chan_list_lock); 759801c1e8dSJohan Hedberg return -EALREADY; 760801c1e8dSJohan Hedberg } 761801c1e8dSJohan Hedberg 762801c1e8dSJohan Hedberg list_add_tail(&c->list, &mgmt_chan_list); 763801c1e8dSJohan Hedberg 764801c1e8dSJohan Hedberg mutex_unlock(&mgmt_chan_list_lock); 765801c1e8dSJohan Hedberg 766801c1e8dSJohan Hedberg return 0; 767801c1e8dSJohan Hedberg } 768801c1e8dSJohan Hedberg EXPORT_SYMBOL(hci_mgmt_chan_register); 769801c1e8dSJohan Hedberg 770801c1e8dSJohan Hedberg void hci_mgmt_chan_unregister(struct hci_mgmt_chan *c) 771801c1e8dSJohan Hedberg { 772801c1e8dSJohan Hedberg mutex_lock(&mgmt_chan_list_lock); 773801c1e8dSJohan Hedberg list_del(&c->list); 774801c1e8dSJohan Hedberg mutex_unlock(&mgmt_chan_list_lock); 775801c1e8dSJohan Hedberg } 776801c1e8dSJohan Hedberg EXPORT_SYMBOL(hci_mgmt_chan_unregister); 777801c1e8dSJohan Hedberg 7781da177e4SLinus Torvalds static int hci_sock_release(struct socket *sock) 7791da177e4SLinus Torvalds { 7801da177e4SLinus Torvalds struct sock *sk = sock->sk; 7817b005bd3SMarcel Holtmann struct hci_dev *hdev; 782249fa169SMarcel Holtmann struct sk_buff *skb; 7831da177e4SLinus Torvalds 7841da177e4SLinus Torvalds BT_DBG("sock %p sk %p", sock, sk); 7851da177e4SLinus Torvalds 7861da177e4SLinus Torvalds if (!sk) 7871da177e4SLinus Torvalds return 0; 7881da177e4SLinus Torvalds 7897b005bd3SMarcel Holtmann hdev = hci_pi(sk)->hdev; 7907b005bd3SMarcel Holtmann 79170ecce91SMarcel Holtmann switch (hci_pi(sk)->channel) { 79270ecce91SMarcel Holtmann case HCI_CHANNEL_MONITOR: 793cd82e61cSMarcel Holtmann atomic_dec(&monitor_promisc); 79470ecce91SMarcel Holtmann break; 79570ecce91SMarcel Holtmann case HCI_CHANNEL_CONTROL: 796249fa169SMarcel Holtmann /* Send event to monitor */ 797249fa169SMarcel Holtmann skb = create_monitor_ctrl_close(sk); 798249fa169SMarcel Holtmann if (skb) { 799249fa169SMarcel Holtmann hci_send_to_channel(HCI_CHANNEL_MONITOR, skb, 800249fa169SMarcel Holtmann HCI_SOCK_TRUSTED, NULL); 801249fa169SMarcel Holtmann kfree_skb(skb); 802249fa169SMarcel Holtmann } 803249fa169SMarcel Holtmann 804df1cb87aSMarcel Holtmann hci_sock_free_cookie(sk); 80570ecce91SMarcel Holtmann break; 80670ecce91SMarcel Holtmann } 807cd82e61cSMarcel Holtmann 8081da177e4SLinus Torvalds bt_sock_unlink(&hci_sk_list, sk); 8091da177e4SLinus Torvalds 8101da177e4SLinus Torvalds if (hdev) { 81123500189SMarcel Holtmann if (hci_pi(sk)->channel == HCI_CHANNEL_USER) { 8126b3cc1dbSSimon Fels /* When releasing an user channel exclusive access, 8136b3cc1dbSSimon Fels * call hci_dev_do_close directly instead of calling 8146b3cc1dbSSimon Fels * hci_dev_close to ensure the exclusive access will 8156b3cc1dbSSimon Fels * be released and the controller brought back down. 8166b3cc1dbSSimon Fels * 8176b3cc1dbSSimon Fels * The checking of HCI_AUTO_OFF is not needed in this 8186b3cc1dbSSimon Fels * case since it will have been cleared already when 8196b3cc1dbSSimon Fels * opening the user channel. 8206b3cc1dbSSimon Fels */ 8216b3cc1dbSSimon Fels hci_dev_do_close(hdev); 8229380f9eaSLoic Poulain hci_dev_clear_flag(hdev, HCI_USER_CHANNEL); 8239380f9eaSLoic Poulain mgmt_index_added(hdev); 82423500189SMarcel Holtmann } 82523500189SMarcel Holtmann 8261da177e4SLinus Torvalds atomic_dec(&hdev->promisc); 8271da177e4SLinus Torvalds hci_dev_put(hdev); 8281da177e4SLinus Torvalds } 8291da177e4SLinus Torvalds 8301da177e4SLinus Torvalds sock_orphan(sk); 8311da177e4SLinus Torvalds 8321da177e4SLinus Torvalds skb_queue_purge(&sk->sk_receive_queue); 8331da177e4SLinus Torvalds skb_queue_purge(&sk->sk_write_queue); 8341da177e4SLinus Torvalds 8351da177e4SLinus Torvalds sock_put(sk); 8361da177e4SLinus Torvalds return 0; 8371da177e4SLinus Torvalds } 8381da177e4SLinus Torvalds 839b2a66aadSAntti Julku static int hci_sock_blacklist_add(struct hci_dev *hdev, void __user *arg) 840f0358568SJohan Hedberg { 841f0358568SJohan Hedberg bdaddr_t bdaddr; 8425e762444SAntti Julku int err; 843f0358568SJohan Hedberg 844f0358568SJohan Hedberg if (copy_from_user(&bdaddr, arg, sizeof(bdaddr))) 845f0358568SJohan Hedberg return -EFAULT; 846f0358568SJohan Hedberg 84709fd0de5SGustavo F. Padovan hci_dev_lock(hdev); 8485e762444SAntti Julku 849dcc36c16SJohan Hedberg err = hci_bdaddr_list_add(&hdev->blacklist, &bdaddr, BDADDR_BREDR); 8505e762444SAntti Julku 85109fd0de5SGustavo F. Padovan hci_dev_unlock(hdev); 8525e762444SAntti Julku 8535e762444SAntti Julku return err; 854f0358568SJohan Hedberg } 855f0358568SJohan Hedberg 856b2a66aadSAntti Julku static int hci_sock_blacklist_del(struct hci_dev *hdev, void __user *arg) 857f0358568SJohan Hedberg { 858f0358568SJohan Hedberg bdaddr_t bdaddr; 8595e762444SAntti Julku int err; 860f0358568SJohan Hedberg 861f0358568SJohan Hedberg if (copy_from_user(&bdaddr, arg, sizeof(bdaddr))) 862f0358568SJohan Hedberg return -EFAULT; 863f0358568SJohan Hedberg 86409fd0de5SGustavo F. Padovan hci_dev_lock(hdev); 8655e762444SAntti Julku 866dcc36c16SJohan Hedberg err = hci_bdaddr_list_del(&hdev->blacklist, &bdaddr, BDADDR_BREDR); 8675e762444SAntti Julku 86809fd0de5SGustavo F. Padovan hci_dev_unlock(hdev); 8695e762444SAntti Julku 8705e762444SAntti Julku return err; 871f0358568SJohan Hedberg } 872f0358568SJohan Hedberg 8731da177e4SLinus Torvalds /* Ioctls that require bound socket */ 8746039aa73SGustavo Padovan static int hci_sock_bound_ioctl(struct sock *sk, unsigned int cmd, 8756039aa73SGustavo Padovan unsigned long arg) 8761da177e4SLinus Torvalds { 8771da177e4SLinus Torvalds struct hci_dev *hdev = hci_pi(sk)->hdev; 8781da177e4SLinus Torvalds 8791da177e4SLinus Torvalds if (!hdev) 8801da177e4SLinus Torvalds return -EBADFD; 8811da177e4SLinus Torvalds 882d7a5a11dSMarcel Holtmann if (hci_dev_test_flag(hdev, HCI_USER_CHANNEL)) 8830736cfa8SMarcel Holtmann return -EBUSY; 8840736cfa8SMarcel Holtmann 885d7a5a11dSMarcel Holtmann if (hci_dev_test_flag(hdev, HCI_UNCONFIGURED)) 886fee746b0SMarcel Holtmann return -EOPNOTSUPP; 887fee746b0SMarcel Holtmann 888ca8bee5dSMarcel Holtmann if (hdev->dev_type != HCI_PRIMARY) 8895b69bef5SMarcel Holtmann return -EOPNOTSUPP; 8905b69bef5SMarcel Holtmann 8911da177e4SLinus Torvalds switch (cmd) { 8921da177e4SLinus Torvalds case HCISETRAW: 8931da177e4SLinus Torvalds if (!capable(CAP_NET_ADMIN)) 894bf5b30b8SZhao Hongjiang return -EPERM; 895db596681SMarcel Holtmann return -EOPNOTSUPP; 8961da177e4SLinus Torvalds 8971da177e4SLinus Torvalds case HCIGETCONNINFO: 8981da177e4SLinus Torvalds return hci_get_conn_info(hdev, (void __user *)arg); 8991da177e4SLinus Torvalds 90040be492fSMarcel Holtmann case HCIGETAUTHINFO: 90140be492fSMarcel Holtmann return hci_get_auth_info(hdev, (void __user *)arg); 90240be492fSMarcel Holtmann 903f0358568SJohan Hedberg case HCIBLOCKADDR: 904f0358568SJohan Hedberg if (!capable(CAP_NET_ADMIN)) 905bf5b30b8SZhao Hongjiang return -EPERM; 906b2a66aadSAntti Julku return hci_sock_blacklist_add(hdev, (void __user *)arg); 907f0358568SJohan Hedberg 908f0358568SJohan Hedberg case HCIUNBLOCKADDR: 909f0358568SJohan Hedberg if (!capable(CAP_NET_ADMIN)) 910bf5b30b8SZhao Hongjiang return -EPERM; 911b2a66aadSAntti Julku return hci_sock_blacklist_del(hdev, (void __user *)arg); 9120736cfa8SMarcel Holtmann } 913f0358568SJohan Hedberg 914324d36edSMarcel Holtmann return -ENOIOCTLCMD; 9151da177e4SLinus Torvalds } 9161da177e4SLinus Torvalds 9178fc9ced3SGustavo Padovan static int hci_sock_ioctl(struct socket *sock, unsigned int cmd, 9188fc9ced3SGustavo Padovan unsigned long arg) 9191da177e4SLinus Torvalds { 9201da177e4SLinus Torvalds void __user *argp = (void __user *)arg; 9210736cfa8SMarcel Holtmann struct sock *sk = sock->sk; 9221da177e4SLinus Torvalds int err; 9231da177e4SLinus Torvalds 9241da177e4SLinus Torvalds BT_DBG("cmd %x arg %lx", cmd, arg); 9251da177e4SLinus Torvalds 926c1c4f956SMarcel Holtmann lock_sock(sk); 927c1c4f956SMarcel Holtmann 928c1c4f956SMarcel Holtmann if (hci_pi(sk)->channel != HCI_CHANNEL_RAW) { 929c1c4f956SMarcel Holtmann err = -EBADFD; 930c1c4f956SMarcel Holtmann goto done; 931c1c4f956SMarcel Holtmann } 932c1c4f956SMarcel Holtmann 933c1c4f956SMarcel Holtmann release_sock(sk); 934c1c4f956SMarcel Holtmann 9351da177e4SLinus Torvalds switch (cmd) { 9361da177e4SLinus Torvalds case HCIGETDEVLIST: 9371da177e4SLinus Torvalds return hci_get_dev_list(argp); 9381da177e4SLinus Torvalds 9391da177e4SLinus Torvalds case HCIGETDEVINFO: 9401da177e4SLinus Torvalds return hci_get_dev_info(argp); 9411da177e4SLinus Torvalds 9421da177e4SLinus Torvalds case HCIGETCONNLIST: 9431da177e4SLinus Torvalds return hci_get_conn_list(argp); 9441da177e4SLinus Torvalds 9451da177e4SLinus Torvalds case HCIDEVUP: 9461da177e4SLinus Torvalds if (!capable(CAP_NET_ADMIN)) 947bf5b30b8SZhao Hongjiang return -EPERM; 9481da177e4SLinus Torvalds return hci_dev_open(arg); 9491da177e4SLinus Torvalds 9501da177e4SLinus Torvalds case HCIDEVDOWN: 9511da177e4SLinus Torvalds if (!capable(CAP_NET_ADMIN)) 952bf5b30b8SZhao Hongjiang return -EPERM; 9531da177e4SLinus Torvalds return hci_dev_close(arg); 9541da177e4SLinus Torvalds 9551da177e4SLinus Torvalds case HCIDEVRESET: 9561da177e4SLinus Torvalds if (!capable(CAP_NET_ADMIN)) 957bf5b30b8SZhao Hongjiang return -EPERM; 9581da177e4SLinus Torvalds return hci_dev_reset(arg); 9591da177e4SLinus Torvalds 9601da177e4SLinus Torvalds case HCIDEVRESTAT: 9611da177e4SLinus Torvalds if (!capable(CAP_NET_ADMIN)) 962bf5b30b8SZhao Hongjiang return -EPERM; 9631da177e4SLinus Torvalds return hci_dev_reset_stat(arg); 9641da177e4SLinus Torvalds 9651da177e4SLinus Torvalds case HCISETSCAN: 9661da177e4SLinus Torvalds case HCISETAUTH: 9671da177e4SLinus Torvalds case HCISETENCRYPT: 9681da177e4SLinus Torvalds case HCISETPTYPE: 9691da177e4SLinus Torvalds case HCISETLINKPOL: 9701da177e4SLinus Torvalds case HCISETLINKMODE: 9711da177e4SLinus Torvalds case HCISETACLMTU: 9721da177e4SLinus Torvalds case HCISETSCOMTU: 9731da177e4SLinus Torvalds if (!capable(CAP_NET_ADMIN)) 974bf5b30b8SZhao Hongjiang return -EPERM; 9751da177e4SLinus Torvalds return hci_dev_cmd(cmd, argp); 9761da177e4SLinus Torvalds 9771da177e4SLinus Torvalds case HCIINQUIRY: 9781da177e4SLinus Torvalds return hci_inquiry(argp); 979c1c4f956SMarcel Holtmann } 9801da177e4SLinus Torvalds 9811da177e4SLinus Torvalds lock_sock(sk); 982c1c4f956SMarcel Holtmann 9831da177e4SLinus Torvalds err = hci_sock_bound_ioctl(sk, cmd, arg); 984c1c4f956SMarcel Holtmann 985c1c4f956SMarcel Holtmann done: 9861da177e4SLinus Torvalds release_sock(sk); 9871da177e4SLinus Torvalds return err; 9881da177e4SLinus Torvalds } 9891da177e4SLinus Torvalds 9908fc9ced3SGustavo Padovan static int hci_sock_bind(struct socket *sock, struct sockaddr *addr, 9918fc9ced3SGustavo Padovan int addr_len) 9921da177e4SLinus Torvalds { 9930381101fSJohan Hedberg struct sockaddr_hci haddr; 9941da177e4SLinus Torvalds struct sock *sk = sock->sk; 9951da177e4SLinus Torvalds struct hci_dev *hdev = NULL; 9960381101fSJohan Hedberg int len, err = 0; 9971da177e4SLinus Torvalds 9981da177e4SLinus Torvalds BT_DBG("sock %p sk %p", sock, sk); 9991da177e4SLinus Torvalds 10000381101fSJohan Hedberg if (!addr) 10010381101fSJohan Hedberg return -EINVAL; 10020381101fSJohan Hedberg 10030381101fSJohan Hedberg memset(&haddr, 0, sizeof(haddr)); 10040381101fSJohan Hedberg len = min_t(unsigned int, sizeof(haddr), addr_len); 10050381101fSJohan Hedberg memcpy(&haddr, addr, len); 10060381101fSJohan Hedberg 10070381101fSJohan Hedberg if (haddr.hci_family != AF_BLUETOOTH) 10080381101fSJohan Hedberg return -EINVAL; 10090381101fSJohan Hedberg 10101da177e4SLinus Torvalds lock_sock(sk); 10111da177e4SLinus Torvalds 10127cc2ade2SMarcel Holtmann if (sk->sk_state == BT_BOUND) { 10137cc2ade2SMarcel Holtmann err = -EALREADY; 10147cc2ade2SMarcel Holtmann goto done; 10157cc2ade2SMarcel Holtmann } 10167cc2ade2SMarcel Holtmann 10177cc2ade2SMarcel Holtmann switch (haddr.hci_channel) { 10187cc2ade2SMarcel Holtmann case HCI_CHANNEL_RAW: 10197cc2ade2SMarcel Holtmann if (hci_pi(sk)->hdev) { 10201da177e4SLinus Torvalds err = -EALREADY; 10211da177e4SLinus Torvalds goto done; 10221da177e4SLinus Torvalds } 10231da177e4SLinus Torvalds 10240381101fSJohan Hedberg if (haddr.hci_dev != HCI_DEV_NONE) { 10250381101fSJohan Hedberg hdev = hci_dev_get(haddr.hci_dev); 102670f23020SAndrei Emeltchenko if (!hdev) { 10271da177e4SLinus Torvalds err = -ENODEV; 10281da177e4SLinus Torvalds goto done; 10291da177e4SLinus Torvalds } 10301da177e4SLinus Torvalds 10311da177e4SLinus Torvalds atomic_inc(&hdev->promisc); 10321da177e4SLinus Torvalds } 10331da177e4SLinus Torvalds 10341da177e4SLinus Torvalds hci_pi(sk)->hdev = hdev; 10357cc2ade2SMarcel Holtmann break; 10367cc2ade2SMarcel Holtmann 103723500189SMarcel Holtmann case HCI_CHANNEL_USER: 103823500189SMarcel Holtmann if (hci_pi(sk)->hdev) { 103923500189SMarcel Holtmann err = -EALREADY; 104023500189SMarcel Holtmann goto done; 104123500189SMarcel Holtmann } 104223500189SMarcel Holtmann 104323500189SMarcel Holtmann if (haddr.hci_dev == HCI_DEV_NONE) { 104423500189SMarcel Holtmann err = -EINVAL; 104523500189SMarcel Holtmann goto done; 104623500189SMarcel Holtmann } 104723500189SMarcel Holtmann 104810a8b86fSMarcel Holtmann if (!capable(CAP_NET_ADMIN)) { 104923500189SMarcel Holtmann err = -EPERM; 105023500189SMarcel Holtmann goto done; 105123500189SMarcel Holtmann } 105223500189SMarcel Holtmann 105323500189SMarcel Holtmann hdev = hci_dev_get(haddr.hci_dev); 105423500189SMarcel Holtmann if (!hdev) { 105523500189SMarcel Holtmann err = -ENODEV; 105623500189SMarcel Holtmann goto done; 105723500189SMarcel Holtmann } 105823500189SMarcel Holtmann 1059781f899fSMarcel Holtmann if (test_bit(HCI_INIT, &hdev->flags) || 1060d7a5a11dSMarcel Holtmann hci_dev_test_flag(hdev, HCI_SETUP) || 1061781f899fSMarcel Holtmann hci_dev_test_flag(hdev, HCI_CONFIG) || 1062781f899fSMarcel Holtmann (!hci_dev_test_flag(hdev, HCI_AUTO_OFF) && 1063781f899fSMarcel Holtmann test_bit(HCI_UP, &hdev->flags))) { 106423500189SMarcel Holtmann err = -EBUSY; 106523500189SMarcel Holtmann hci_dev_put(hdev); 106623500189SMarcel Holtmann goto done; 106723500189SMarcel Holtmann } 106823500189SMarcel Holtmann 1069238be788SMarcel Holtmann if (hci_dev_test_and_set_flag(hdev, HCI_USER_CHANNEL)) { 107023500189SMarcel Holtmann err = -EUSERS; 107123500189SMarcel Holtmann hci_dev_put(hdev); 107223500189SMarcel Holtmann goto done; 107323500189SMarcel Holtmann } 107423500189SMarcel Holtmann 107523500189SMarcel Holtmann mgmt_index_removed(hdev); 107623500189SMarcel Holtmann 107723500189SMarcel Holtmann err = hci_dev_open(hdev->id); 107823500189SMarcel Holtmann if (err) { 1079781f899fSMarcel Holtmann if (err == -EALREADY) { 1080781f899fSMarcel Holtmann /* In case the transport is already up and 1081781f899fSMarcel Holtmann * running, clear the error here. 1082781f899fSMarcel Holtmann * 1083781f899fSMarcel Holtmann * This can happen when opening an user 1084781f899fSMarcel Holtmann * channel and HCI_AUTO_OFF grace period 1085781f899fSMarcel Holtmann * is still active. 1086781f899fSMarcel Holtmann */ 1087781f899fSMarcel Holtmann err = 0; 1088781f899fSMarcel Holtmann } else { 1089a358dc11SMarcel Holtmann hci_dev_clear_flag(hdev, HCI_USER_CHANNEL); 1090c6521401SMarcel Holtmann mgmt_index_added(hdev); 109123500189SMarcel Holtmann hci_dev_put(hdev); 109223500189SMarcel Holtmann goto done; 109323500189SMarcel Holtmann } 1094781f899fSMarcel Holtmann } 109523500189SMarcel Holtmann 109623500189SMarcel Holtmann atomic_inc(&hdev->promisc); 109723500189SMarcel Holtmann 109823500189SMarcel Holtmann hci_pi(sk)->hdev = hdev; 109923500189SMarcel Holtmann break; 110023500189SMarcel Holtmann 1101cd82e61cSMarcel Holtmann case HCI_CHANNEL_MONITOR: 1102cd82e61cSMarcel Holtmann if (haddr.hci_dev != HCI_DEV_NONE) { 1103cd82e61cSMarcel Holtmann err = -EINVAL; 1104cd82e61cSMarcel Holtmann goto done; 1105cd82e61cSMarcel Holtmann } 1106cd82e61cSMarcel Holtmann 1107cd82e61cSMarcel Holtmann if (!capable(CAP_NET_RAW)) { 1108cd82e61cSMarcel Holtmann err = -EPERM; 1109cd82e61cSMarcel Holtmann goto done; 1110cd82e61cSMarcel Holtmann } 1111cd82e61cSMarcel Holtmann 111250ebc055SMarcel Holtmann /* The monitor interface is restricted to CAP_NET_RAW 111350ebc055SMarcel Holtmann * capabilities and with that implicitly trusted. 111450ebc055SMarcel Holtmann */ 111550ebc055SMarcel Holtmann hci_sock_set_flag(sk, HCI_SOCK_TRUSTED); 111650ebc055SMarcel Holtmann 1117787b306cSJohannes Berg send_monitor_note(sk, "Linux version %s (%s)", 1118787b306cSJohannes Berg init_utsname()->release, 1119787b306cSJohannes Berg init_utsname()->machine); 1120*9e8305b3SMarcel Holtmann send_monitor_note(sk, "Bluetooth subsystem version %u.%u", 1121*9e8305b3SMarcel Holtmann BT_SUBSYS_VERSION, BT_SUBSYS_REVISION); 1122cd82e61cSMarcel Holtmann send_monitor_replay(sk); 1123249fa169SMarcel Holtmann send_monitor_control_replay(sk); 1124cd82e61cSMarcel Holtmann 1125cd82e61cSMarcel Holtmann atomic_inc(&monitor_promisc); 1126cd82e61cSMarcel Holtmann break; 1127cd82e61cSMarcel Holtmann 1128ac714949SMarcel Holtmann case HCI_CHANNEL_LOGGING: 1129ac714949SMarcel Holtmann if (haddr.hci_dev != HCI_DEV_NONE) { 1130ac714949SMarcel Holtmann err = -EINVAL; 1131ac714949SMarcel Holtmann goto done; 1132ac714949SMarcel Holtmann } 1133ac714949SMarcel Holtmann 1134ac714949SMarcel Holtmann if (!capable(CAP_NET_ADMIN)) { 1135ac714949SMarcel Holtmann err = -EPERM; 1136ac714949SMarcel Holtmann goto done; 1137ac714949SMarcel Holtmann } 1138ac714949SMarcel Holtmann break; 1139ac714949SMarcel Holtmann 11407cc2ade2SMarcel Holtmann default: 1141801c1e8dSJohan Hedberg if (!hci_mgmt_chan_find(haddr.hci_channel)) { 11427cc2ade2SMarcel Holtmann err = -EINVAL; 11437cc2ade2SMarcel Holtmann goto done; 11447cc2ade2SMarcel Holtmann } 11457cc2ade2SMarcel Holtmann 1146801c1e8dSJohan Hedberg if (haddr.hci_dev != HCI_DEV_NONE) { 1147801c1e8dSJohan Hedberg err = -EINVAL; 1148801c1e8dSJohan Hedberg goto done; 1149801c1e8dSJohan Hedberg } 1150801c1e8dSJohan Hedberg 11511195fbb8SMarcel Holtmann /* Users with CAP_NET_ADMIN capabilities are allowed 11521195fbb8SMarcel Holtmann * access to all management commands and events. For 11531195fbb8SMarcel Holtmann * untrusted users the interface is restricted and 11541195fbb8SMarcel Holtmann * also only untrusted events are sent. 115550ebc055SMarcel Holtmann */ 11561195fbb8SMarcel Holtmann if (capable(CAP_NET_ADMIN)) 115750ebc055SMarcel Holtmann hci_sock_set_flag(sk, HCI_SOCK_TRUSTED); 115850ebc055SMarcel Holtmann 1159f9207338SMarcel Holtmann /* At the moment the index and unconfigured index events 1160f9207338SMarcel Holtmann * are enabled unconditionally. Setting them on each 1161f9207338SMarcel Holtmann * socket when binding keeps this functionality. They 1162f9207338SMarcel Holtmann * however might be cleared later and then sending of these 1163f9207338SMarcel Holtmann * events will be disabled, but that is then intentional. 1164f6b7712eSMarcel Holtmann * 1165f6b7712eSMarcel Holtmann * This also enables generic events that are safe to be 1166f6b7712eSMarcel Holtmann * received by untrusted users. Example for such events 1167f6b7712eSMarcel Holtmann * are changes to settings, class of device, name etc. 1168f9207338SMarcel Holtmann */ 1169f9207338SMarcel Holtmann if (haddr.hci_channel == HCI_CHANNEL_CONTROL) { 1170249fa169SMarcel Holtmann struct sk_buff *skb; 117170ecce91SMarcel Holtmann 1172df1cb87aSMarcel Holtmann hci_sock_gen_cookie(sk); 117370ecce91SMarcel Holtmann 1174249fa169SMarcel Holtmann /* Send event to monitor */ 1175249fa169SMarcel Holtmann skb = create_monitor_ctrl_open(sk); 1176249fa169SMarcel Holtmann if (skb) { 1177249fa169SMarcel Holtmann hci_send_to_channel(HCI_CHANNEL_MONITOR, skb, 1178249fa169SMarcel Holtmann HCI_SOCK_TRUSTED, NULL); 1179249fa169SMarcel Holtmann kfree_skb(skb); 1180249fa169SMarcel Holtmann } 1181249fa169SMarcel Holtmann 1182f9207338SMarcel Holtmann hci_sock_set_flag(sk, HCI_MGMT_INDEX_EVENTS); 1183f9207338SMarcel Holtmann hci_sock_set_flag(sk, HCI_MGMT_UNCONF_INDEX_EVENTS); 11845504c3a3SMarcel Holtmann hci_sock_set_flag(sk, HCI_MGMT_OPTION_EVENTS); 11855504c3a3SMarcel Holtmann hci_sock_set_flag(sk, HCI_MGMT_SETTING_EVENTS); 11865504c3a3SMarcel Holtmann hci_sock_set_flag(sk, HCI_MGMT_DEV_CLASS_EVENTS); 11875504c3a3SMarcel Holtmann hci_sock_set_flag(sk, HCI_MGMT_LOCAL_NAME_EVENTS); 1188f9207338SMarcel Holtmann } 1189801c1e8dSJohan Hedberg break; 1190801c1e8dSJohan Hedberg } 1191801c1e8dSJohan Hedberg 11927cc2ade2SMarcel Holtmann 11937cc2ade2SMarcel Holtmann hci_pi(sk)->channel = haddr.hci_channel; 11941da177e4SLinus Torvalds sk->sk_state = BT_BOUND; 11951da177e4SLinus Torvalds 11961da177e4SLinus Torvalds done: 11971da177e4SLinus Torvalds release_sock(sk); 11981da177e4SLinus Torvalds return err; 11991da177e4SLinus Torvalds } 12001da177e4SLinus Torvalds 12018fc9ced3SGustavo Padovan static int hci_sock_getname(struct socket *sock, struct sockaddr *addr, 12028fc9ced3SGustavo Padovan int *addr_len, int peer) 12031da177e4SLinus Torvalds { 12041da177e4SLinus Torvalds struct sockaddr_hci *haddr = (struct sockaddr_hci *)addr; 12051da177e4SLinus Torvalds struct sock *sk = sock->sk; 12069d4b68b2SMarcel Holtmann struct hci_dev *hdev; 12079d4b68b2SMarcel Holtmann int err = 0; 12081da177e4SLinus Torvalds 12091da177e4SLinus Torvalds BT_DBG("sock %p sk %p", sock, sk); 12101da177e4SLinus Torvalds 121106f43cbcSMarcel Holtmann if (peer) 121206f43cbcSMarcel Holtmann return -EOPNOTSUPP; 121306f43cbcSMarcel Holtmann 12141da177e4SLinus Torvalds lock_sock(sk); 12151da177e4SLinus Torvalds 12169d4b68b2SMarcel Holtmann hdev = hci_pi(sk)->hdev; 12179d4b68b2SMarcel Holtmann if (!hdev) { 12189d4b68b2SMarcel Holtmann err = -EBADFD; 12199d4b68b2SMarcel Holtmann goto done; 12209d4b68b2SMarcel Holtmann } 12219d4b68b2SMarcel Holtmann 12221da177e4SLinus Torvalds *addr_len = sizeof(*haddr); 12231da177e4SLinus Torvalds haddr->hci_family = AF_BLUETOOTH; 12247b005bd3SMarcel Holtmann haddr->hci_dev = hdev->id; 12259d4b68b2SMarcel Holtmann haddr->hci_channel= hci_pi(sk)->channel; 12261da177e4SLinus Torvalds 12279d4b68b2SMarcel Holtmann done: 12281da177e4SLinus Torvalds release_sock(sk); 12299d4b68b2SMarcel Holtmann return err; 12301da177e4SLinus Torvalds } 12311da177e4SLinus Torvalds 12326039aa73SGustavo Padovan static void hci_sock_cmsg(struct sock *sk, struct msghdr *msg, 12336039aa73SGustavo Padovan struct sk_buff *skb) 12341da177e4SLinus Torvalds { 12351da177e4SLinus Torvalds __u32 mask = hci_pi(sk)->cmsg_mask; 12361da177e4SLinus Torvalds 12370d48d939SMarcel Holtmann if (mask & HCI_CMSG_DIR) { 12380d48d939SMarcel Holtmann int incoming = bt_cb(skb)->incoming; 12398fc9ced3SGustavo Padovan put_cmsg(msg, SOL_HCI, HCI_CMSG_DIR, sizeof(incoming), 12408fc9ced3SGustavo Padovan &incoming); 12410d48d939SMarcel Holtmann } 12421da177e4SLinus Torvalds 1243a61bbcf2SPatrick McHardy if (mask & HCI_CMSG_TSTAMP) { 1244f6e623a6SJohann Felix Soden #ifdef CONFIG_COMPAT 1245f6e623a6SJohann Felix Soden struct compat_timeval ctv; 1246f6e623a6SJohann Felix Soden #endif 1247a61bbcf2SPatrick McHardy struct timeval tv; 1248767c5eb5SMarcel Holtmann void *data; 1249767c5eb5SMarcel Holtmann int len; 1250a61bbcf2SPatrick McHardy 1251a61bbcf2SPatrick McHardy skb_get_timestamp(skb, &tv); 1252767c5eb5SMarcel Holtmann 12531da97f83SDavid S. Miller data = &tv; 12541da97f83SDavid S. Miller len = sizeof(tv); 12551da97f83SDavid S. Miller #ifdef CONFIG_COMPAT 1256da88cea1SH. J. Lu if (!COMPAT_USE_64BIT_TIME && 1257da88cea1SH. J. Lu (msg->msg_flags & MSG_CMSG_COMPAT)) { 1258767c5eb5SMarcel Holtmann ctv.tv_sec = tv.tv_sec; 1259767c5eb5SMarcel Holtmann ctv.tv_usec = tv.tv_usec; 1260767c5eb5SMarcel Holtmann data = &ctv; 1261767c5eb5SMarcel Holtmann len = sizeof(ctv); 1262767c5eb5SMarcel Holtmann } 12631da97f83SDavid S. Miller #endif 1264767c5eb5SMarcel Holtmann 1265767c5eb5SMarcel Holtmann put_cmsg(msg, SOL_HCI, HCI_CMSG_TSTAMP, len, data); 1266a61bbcf2SPatrick McHardy } 12671da177e4SLinus Torvalds } 12681da177e4SLinus Torvalds 12698528d3f7SMarcel Holtmann static int hci_sock_recvmsg(struct socket *sock, struct msghdr *msg, 12708528d3f7SMarcel Holtmann size_t len, int flags) 12711da177e4SLinus Torvalds { 12721da177e4SLinus Torvalds int noblock = flags & MSG_DONTWAIT; 12731da177e4SLinus Torvalds struct sock *sk = sock->sk; 12741da177e4SLinus Torvalds struct sk_buff *skb; 12751da177e4SLinus Torvalds int copied, err; 127683871f8cSDenis Kenzior unsigned int skblen; 12771da177e4SLinus Torvalds 12781da177e4SLinus Torvalds BT_DBG("sock %p, sk %p", sock, sk); 12791da177e4SLinus Torvalds 1280d94a6104SMarcel Holtmann if (flags & MSG_OOB) 12811da177e4SLinus Torvalds return -EOPNOTSUPP; 12821da177e4SLinus Torvalds 1283ac714949SMarcel Holtmann if (hci_pi(sk)->channel == HCI_CHANNEL_LOGGING) 1284ac714949SMarcel Holtmann return -EOPNOTSUPP; 1285ac714949SMarcel Holtmann 12861da177e4SLinus Torvalds if (sk->sk_state == BT_CLOSED) 12871da177e4SLinus Torvalds return 0; 12881da177e4SLinus Torvalds 128970f23020SAndrei Emeltchenko skb = skb_recv_datagram(sk, flags, noblock, &err); 129070f23020SAndrei Emeltchenko if (!skb) 12911da177e4SLinus Torvalds return err; 12921da177e4SLinus Torvalds 129383871f8cSDenis Kenzior skblen = skb->len; 12941da177e4SLinus Torvalds copied = skb->len; 12951da177e4SLinus Torvalds if (len < copied) { 12961da177e4SLinus Torvalds msg->msg_flags |= MSG_TRUNC; 12971da177e4SLinus Torvalds copied = len; 12981da177e4SLinus Torvalds } 12991da177e4SLinus Torvalds 1300badff6d0SArnaldo Carvalho de Melo skb_reset_transport_header(skb); 130151f3d02bSDavid S. Miller err = skb_copy_datagram_msg(skb, 0, msg, copied); 13021da177e4SLinus Torvalds 13033a208627SMarcel Holtmann switch (hci_pi(sk)->channel) { 13043a208627SMarcel Holtmann case HCI_CHANNEL_RAW: 13051da177e4SLinus Torvalds hci_sock_cmsg(sk, msg, skb); 13063a208627SMarcel Holtmann break; 130723500189SMarcel Holtmann case HCI_CHANNEL_USER: 1308cd82e61cSMarcel Holtmann case HCI_CHANNEL_MONITOR: 1309cd82e61cSMarcel Holtmann sock_recv_timestamp(msg, sk, skb); 1310cd82e61cSMarcel Holtmann break; 1311801c1e8dSJohan Hedberg default: 1312801c1e8dSJohan Hedberg if (hci_mgmt_chan_find(hci_pi(sk)->channel)) 1313801c1e8dSJohan Hedberg sock_recv_timestamp(msg, sk, skb); 1314801c1e8dSJohan Hedberg break; 13153a208627SMarcel Holtmann } 13161da177e4SLinus Torvalds 13171da177e4SLinus Torvalds skb_free_datagram(sk, skb); 13181da177e4SLinus Torvalds 13194f34228bSLuiz Augusto von Dentz if (flags & MSG_TRUNC) 132083871f8cSDenis Kenzior copied = skblen; 132183871f8cSDenis Kenzior 13221da177e4SLinus Torvalds return err ? : copied; 13231da177e4SLinus Torvalds } 13241da177e4SLinus Torvalds 1325fa4335d7SJohan Hedberg static int hci_mgmt_cmd(struct hci_mgmt_chan *chan, struct sock *sk, 1326fa4335d7SJohan Hedberg struct msghdr *msg, size_t msglen) 1327fa4335d7SJohan Hedberg { 1328fa4335d7SJohan Hedberg void *buf; 1329fa4335d7SJohan Hedberg u8 *cp; 1330fa4335d7SJohan Hedberg struct mgmt_hdr *hdr; 1331fa4335d7SJohan Hedberg u16 opcode, index, len; 1332fa4335d7SJohan Hedberg struct hci_dev *hdev = NULL; 1333fa4335d7SJohan Hedberg const struct hci_mgmt_handler *handler; 1334fa4335d7SJohan Hedberg bool var_len, no_hdev; 1335fa4335d7SJohan Hedberg int err; 1336fa4335d7SJohan Hedberg 1337fa4335d7SJohan Hedberg BT_DBG("got %zu bytes", msglen); 1338fa4335d7SJohan Hedberg 1339fa4335d7SJohan Hedberg if (msglen < sizeof(*hdr)) 1340fa4335d7SJohan Hedberg return -EINVAL; 1341fa4335d7SJohan Hedberg 1342fa4335d7SJohan Hedberg buf = kmalloc(msglen, GFP_KERNEL); 1343fa4335d7SJohan Hedberg if (!buf) 1344fa4335d7SJohan Hedberg return -ENOMEM; 1345fa4335d7SJohan Hedberg 1346fa4335d7SJohan Hedberg if (memcpy_from_msg(buf, msg, msglen)) { 1347fa4335d7SJohan Hedberg err = -EFAULT; 1348fa4335d7SJohan Hedberg goto done; 1349fa4335d7SJohan Hedberg } 1350fa4335d7SJohan Hedberg 1351fa4335d7SJohan Hedberg hdr = buf; 1352fa4335d7SJohan Hedberg opcode = __le16_to_cpu(hdr->opcode); 1353fa4335d7SJohan Hedberg index = __le16_to_cpu(hdr->index); 1354fa4335d7SJohan Hedberg len = __le16_to_cpu(hdr->len); 1355fa4335d7SJohan Hedberg 1356fa4335d7SJohan Hedberg if (len != msglen - sizeof(*hdr)) { 1357fa4335d7SJohan Hedberg err = -EINVAL; 1358fa4335d7SJohan Hedberg goto done; 1359fa4335d7SJohan Hedberg } 1360fa4335d7SJohan Hedberg 136138ceaa00SMarcel Holtmann if (chan->channel == HCI_CHANNEL_CONTROL) { 136238ceaa00SMarcel Holtmann struct sk_buff *skb; 136338ceaa00SMarcel Holtmann 136438ceaa00SMarcel Holtmann /* Send event to monitor */ 136538ceaa00SMarcel Holtmann skb = create_monitor_ctrl_command(sk, index, opcode, len, 136638ceaa00SMarcel Holtmann buf + sizeof(*hdr)); 136738ceaa00SMarcel Holtmann if (skb) { 136838ceaa00SMarcel Holtmann hci_send_to_channel(HCI_CHANNEL_MONITOR, skb, 136938ceaa00SMarcel Holtmann HCI_SOCK_TRUSTED, NULL); 137038ceaa00SMarcel Holtmann kfree_skb(skb); 137138ceaa00SMarcel Holtmann } 137238ceaa00SMarcel Holtmann } 137338ceaa00SMarcel Holtmann 1374fa4335d7SJohan Hedberg if (opcode >= chan->handler_count || 1375fa4335d7SJohan Hedberg chan->handlers[opcode].func == NULL) { 1376fa4335d7SJohan Hedberg BT_DBG("Unknown op %u", opcode); 1377fa4335d7SJohan Hedberg err = mgmt_cmd_status(sk, index, opcode, 1378fa4335d7SJohan Hedberg MGMT_STATUS_UNKNOWN_COMMAND); 1379fa4335d7SJohan Hedberg goto done; 1380fa4335d7SJohan Hedberg } 1381fa4335d7SJohan Hedberg 1382fa4335d7SJohan Hedberg handler = &chan->handlers[opcode]; 1383fa4335d7SJohan Hedberg 1384fa4335d7SJohan Hedberg if (!hci_sock_test_flag(sk, HCI_SOCK_TRUSTED) && 1385fa4335d7SJohan Hedberg !(handler->flags & HCI_MGMT_UNTRUSTED)) { 1386fa4335d7SJohan Hedberg err = mgmt_cmd_status(sk, index, opcode, 1387fa4335d7SJohan Hedberg MGMT_STATUS_PERMISSION_DENIED); 1388fa4335d7SJohan Hedberg goto done; 1389fa4335d7SJohan Hedberg } 1390fa4335d7SJohan Hedberg 1391fa4335d7SJohan Hedberg if (index != MGMT_INDEX_NONE) { 1392fa4335d7SJohan Hedberg hdev = hci_dev_get(index); 1393fa4335d7SJohan Hedberg if (!hdev) { 1394fa4335d7SJohan Hedberg err = mgmt_cmd_status(sk, index, opcode, 1395fa4335d7SJohan Hedberg MGMT_STATUS_INVALID_INDEX); 1396fa4335d7SJohan Hedberg goto done; 1397fa4335d7SJohan Hedberg } 1398fa4335d7SJohan Hedberg 1399fa4335d7SJohan Hedberg if (hci_dev_test_flag(hdev, HCI_SETUP) || 1400fa4335d7SJohan Hedberg hci_dev_test_flag(hdev, HCI_CONFIG) || 1401fa4335d7SJohan Hedberg hci_dev_test_flag(hdev, HCI_USER_CHANNEL)) { 1402fa4335d7SJohan Hedberg err = mgmt_cmd_status(sk, index, opcode, 1403fa4335d7SJohan Hedberg MGMT_STATUS_INVALID_INDEX); 1404fa4335d7SJohan Hedberg goto done; 1405fa4335d7SJohan Hedberg } 1406fa4335d7SJohan Hedberg 1407fa4335d7SJohan Hedberg if (hci_dev_test_flag(hdev, HCI_UNCONFIGURED) && 1408fa4335d7SJohan Hedberg !(handler->flags & HCI_MGMT_UNCONFIGURED)) { 1409fa4335d7SJohan Hedberg err = mgmt_cmd_status(sk, index, opcode, 1410fa4335d7SJohan Hedberg MGMT_STATUS_INVALID_INDEX); 1411fa4335d7SJohan Hedberg goto done; 1412fa4335d7SJohan Hedberg } 1413fa4335d7SJohan Hedberg } 1414fa4335d7SJohan Hedberg 1415fa4335d7SJohan Hedberg no_hdev = (handler->flags & HCI_MGMT_NO_HDEV); 1416fa4335d7SJohan Hedberg if (no_hdev != !hdev) { 1417fa4335d7SJohan Hedberg err = mgmt_cmd_status(sk, index, opcode, 1418fa4335d7SJohan Hedberg MGMT_STATUS_INVALID_INDEX); 1419fa4335d7SJohan Hedberg goto done; 1420fa4335d7SJohan Hedberg } 1421fa4335d7SJohan Hedberg 1422fa4335d7SJohan Hedberg var_len = (handler->flags & HCI_MGMT_VAR_LEN); 1423fa4335d7SJohan Hedberg if ((var_len && len < handler->data_len) || 1424fa4335d7SJohan Hedberg (!var_len && len != handler->data_len)) { 1425fa4335d7SJohan Hedberg err = mgmt_cmd_status(sk, index, opcode, 1426fa4335d7SJohan Hedberg MGMT_STATUS_INVALID_PARAMS); 1427fa4335d7SJohan Hedberg goto done; 1428fa4335d7SJohan Hedberg } 1429fa4335d7SJohan Hedberg 1430fa4335d7SJohan Hedberg if (hdev && chan->hdev_init) 1431fa4335d7SJohan Hedberg chan->hdev_init(sk, hdev); 1432fa4335d7SJohan Hedberg 1433fa4335d7SJohan Hedberg cp = buf + sizeof(*hdr); 1434fa4335d7SJohan Hedberg 1435fa4335d7SJohan Hedberg err = handler->func(sk, hdev, cp, len); 1436fa4335d7SJohan Hedberg if (err < 0) 1437fa4335d7SJohan Hedberg goto done; 1438fa4335d7SJohan Hedberg 1439fa4335d7SJohan Hedberg err = msglen; 1440fa4335d7SJohan Hedberg 1441fa4335d7SJohan Hedberg done: 1442fa4335d7SJohan Hedberg if (hdev) 1443fa4335d7SJohan Hedberg hci_dev_put(hdev); 1444fa4335d7SJohan Hedberg 1445fa4335d7SJohan Hedberg kfree(buf); 1446fa4335d7SJohan Hedberg return err; 1447fa4335d7SJohan Hedberg } 1448fa4335d7SJohan Hedberg 1449ac714949SMarcel Holtmann static int hci_logging_frame(struct sock *sk, struct msghdr *msg, int len) 1450ac714949SMarcel Holtmann { 1451ac714949SMarcel Holtmann struct hci_mon_hdr *hdr; 1452ac714949SMarcel Holtmann struct sk_buff *skb; 1453ac714949SMarcel Holtmann struct hci_dev *hdev; 1454ac714949SMarcel Holtmann u16 index; 1455ac714949SMarcel Holtmann int err; 1456ac714949SMarcel Holtmann 1457ac714949SMarcel Holtmann /* The logging frame consists at minimum of the standard header, 1458ac714949SMarcel Holtmann * the priority byte, the ident length byte and at least one string 1459ac714949SMarcel Holtmann * terminator NUL byte. Anything shorter are invalid packets. 1460ac714949SMarcel Holtmann */ 1461ac714949SMarcel Holtmann if (len < sizeof(*hdr) + 3) 1462ac714949SMarcel Holtmann return -EINVAL; 1463ac714949SMarcel Holtmann 1464ac714949SMarcel Holtmann skb = bt_skb_send_alloc(sk, len, msg->msg_flags & MSG_DONTWAIT, &err); 1465ac714949SMarcel Holtmann if (!skb) 1466ac714949SMarcel Holtmann return err; 1467ac714949SMarcel Holtmann 1468ac714949SMarcel Holtmann if (memcpy_from_msg(skb_put(skb, len), msg, len)) { 1469ac714949SMarcel Holtmann err = -EFAULT; 1470ac714949SMarcel Holtmann goto drop; 1471ac714949SMarcel Holtmann } 1472ac714949SMarcel Holtmann 1473ac714949SMarcel Holtmann hdr = (void *)skb->data; 1474ac714949SMarcel Holtmann 1475ac714949SMarcel Holtmann if (__le16_to_cpu(hdr->len) != len - sizeof(*hdr)) { 1476ac714949SMarcel Holtmann err = -EINVAL; 1477ac714949SMarcel Holtmann goto drop; 1478ac714949SMarcel Holtmann } 1479ac714949SMarcel Holtmann 1480ac714949SMarcel Holtmann if (__le16_to_cpu(hdr->opcode) == 0x0000) { 1481ac714949SMarcel Holtmann __u8 priority = skb->data[sizeof(*hdr)]; 1482ac714949SMarcel Holtmann __u8 ident_len = skb->data[sizeof(*hdr) + 1]; 1483ac714949SMarcel Holtmann 1484ac714949SMarcel Holtmann /* Only the priorities 0-7 are valid and with that any other 1485ac714949SMarcel Holtmann * value results in an invalid packet. 1486ac714949SMarcel Holtmann * 1487ac714949SMarcel Holtmann * The priority byte is followed by an ident length byte and 1488ac714949SMarcel Holtmann * the NUL terminated ident string. Check that the ident 1489ac714949SMarcel Holtmann * length is not overflowing the packet and also that the 1490ac714949SMarcel Holtmann * ident string itself is NUL terminated. In case the ident 1491ac714949SMarcel Holtmann * length is zero, the length value actually doubles as NUL 1492ac714949SMarcel Holtmann * terminator identifier. 1493ac714949SMarcel Holtmann * 1494ac714949SMarcel Holtmann * The message follows the ident string (if present) and 1495ac714949SMarcel Holtmann * must be NUL terminated. Otherwise it is not a valid packet. 1496ac714949SMarcel Holtmann */ 1497ac714949SMarcel Holtmann if (priority > 7 || skb->data[len - 1] != 0x00 || 1498ac714949SMarcel Holtmann ident_len > len - sizeof(*hdr) - 3 || 1499ac714949SMarcel Holtmann skb->data[sizeof(*hdr) + ident_len + 1] != 0x00) { 1500ac714949SMarcel Holtmann err = -EINVAL; 1501ac714949SMarcel Holtmann goto drop; 1502ac714949SMarcel Holtmann } 1503ac714949SMarcel Holtmann } else { 1504ac714949SMarcel Holtmann err = -EINVAL; 1505ac714949SMarcel Holtmann goto drop; 1506ac714949SMarcel Holtmann } 1507ac714949SMarcel Holtmann 1508ac714949SMarcel Holtmann index = __le16_to_cpu(hdr->index); 1509ac714949SMarcel Holtmann 1510ac714949SMarcel Holtmann if (index != MGMT_INDEX_NONE) { 1511ac714949SMarcel Holtmann hdev = hci_dev_get(index); 1512ac714949SMarcel Holtmann if (!hdev) { 1513ac714949SMarcel Holtmann err = -ENODEV; 1514ac714949SMarcel Holtmann goto drop; 1515ac714949SMarcel Holtmann } 1516ac714949SMarcel Holtmann } else { 1517ac714949SMarcel Holtmann hdev = NULL; 1518ac714949SMarcel Holtmann } 1519ac714949SMarcel Holtmann 1520ac714949SMarcel Holtmann hdr->opcode = cpu_to_le16(HCI_MON_USER_LOGGING); 1521ac714949SMarcel Holtmann 1522ac714949SMarcel Holtmann hci_send_to_channel(HCI_CHANNEL_MONITOR, skb, HCI_SOCK_TRUSTED, NULL); 1523ac714949SMarcel Holtmann err = len; 1524ac714949SMarcel Holtmann 1525ac714949SMarcel Holtmann if (hdev) 1526ac714949SMarcel Holtmann hci_dev_put(hdev); 1527ac714949SMarcel Holtmann 1528ac714949SMarcel Holtmann drop: 1529ac714949SMarcel Holtmann kfree_skb(skb); 1530ac714949SMarcel Holtmann return err; 1531ac714949SMarcel Holtmann } 1532ac714949SMarcel Holtmann 15331b784140SYing Xue static int hci_sock_sendmsg(struct socket *sock, struct msghdr *msg, 15341b784140SYing Xue size_t len) 15351da177e4SLinus Torvalds { 15361da177e4SLinus Torvalds struct sock *sk = sock->sk; 1537801c1e8dSJohan Hedberg struct hci_mgmt_chan *chan; 15381da177e4SLinus Torvalds struct hci_dev *hdev; 15391da177e4SLinus Torvalds struct sk_buff *skb; 15401da177e4SLinus Torvalds int err; 15411da177e4SLinus Torvalds 15421da177e4SLinus Torvalds BT_DBG("sock %p sk %p", sock, sk); 15431da177e4SLinus Torvalds 15441da177e4SLinus Torvalds if (msg->msg_flags & MSG_OOB) 15451da177e4SLinus Torvalds return -EOPNOTSUPP; 15461da177e4SLinus Torvalds 15471da177e4SLinus Torvalds if (msg->msg_flags & ~(MSG_DONTWAIT|MSG_NOSIGNAL|MSG_ERRQUEUE)) 15481da177e4SLinus Torvalds return -EINVAL; 15491da177e4SLinus Torvalds 15501da177e4SLinus Torvalds if (len < 4 || len > HCI_MAX_FRAME_SIZE) 15511da177e4SLinus Torvalds return -EINVAL; 15521da177e4SLinus Torvalds 15531da177e4SLinus Torvalds lock_sock(sk); 15541da177e4SLinus Torvalds 15550381101fSJohan Hedberg switch (hci_pi(sk)->channel) { 15560381101fSJohan Hedberg case HCI_CHANNEL_RAW: 155723500189SMarcel Holtmann case HCI_CHANNEL_USER: 15580381101fSJohan Hedberg break; 1559cd82e61cSMarcel Holtmann case HCI_CHANNEL_MONITOR: 1560cd82e61cSMarcel Holtmann err = -EOPNOTSUPP; 1561cd82e61cSMarcel Holtmann goto done; 1562ac714949SMarcel Holtmann case HCI_CHANNEL_LOGGING: 1563ac714949SMarcel Holtmann err = hci_logging_frame(sk, msg, len); 1564ac714949SMarcel Holtmann goto done; 15650381101fSJohan Hedberg default: 1566801c1e8dSJohan Hedberg mutex_lock(&mgmt_chan_list_lock); 1567801c1e8dSJohan Hedberg chan = __hci_mgmt_chan_find(hci_pi(sk)->channel); 1568801c1e8dSJohan Hedberg if (chan) 1569fa4335d7SJohan Hedberg err = hci_mgmt_cmd(chan, sk, msg, len); 1570801c1e8dSJohan Hedberg else 15710381101fSJohan Hedberg err = -EINVAL; 1572801c1e8dSJohan Hedberg 1573801c1e8dSJohan Hedberg mutex_unlock(&mgmt_chan_list_lock); 15740381101fSJohan Hedberg goto done; 15750381101fSJohan Hedberg } 15760381101fSJohan Hedberg 157770f23020SAndrei Emeltchenko hdev = hci_pi(sk)->hdev; 157870f23020SAndrei Emeltchenko if (!hdev) { 15791da177e4SLinus Torvalds err = -EBADFD; 15801da177e4SLinus Torvalds goto done; 15811da177e4SLinus Torvalds } 15821da177e4SLinus Torvalds 15837e21addcSMarcel Holtmann if (!test_bit(HCI_UP, &hdev->flags)) { 15847e21addcSMarcel Holtmann err = -ENETDOWN; 15857e21addcSMarcel Holtmann goto done; 15867e21addcSMarcel Holtmann } 15877e21addcSMarcel Holtmann 158870f23020SAndrei Emeltchenko skb = bt_skb_send_alloc(sk, len, msg->msg_flags & MSG_DONTWAIT, &err); 158970f23020SAndrei Emeltchenko if (!skb) 15901da177e4SLinus Torvalds goto done; 15911da177e4SLinus Torvalds 15926ce8e9ceSAl Viro if (memcpy_from_msg(skb_put(skb, len), msg, len)) { 15931da177e4SLinus Torvalds err = -EFAULT; 15941da177e4SLinus Torvalds goto drop; 15951da177e4SLinus Torvalds } 15961da177e4SLinus Torvalds 15978528d3f7SMarcel Holtmann hci_skb_pkt_type(skb) = skb->data[0]; 15981da177e4SLinus Torvalds skb_pull(skb, 1); 15991da177e4SLinus Torvalds 16001bc5ad16SMarcel Holtmann if (hci_pi(sk)->channel == HCI_CHANNEL_USER) { 16011bc5ad16SMarcel Holtmann /* No permission check is needed for user channel 16021bc5ad16SMarcel Holtmann * since that gets enforced when binding the socket. 16031bc5ad16SMarcel Holtmann * 16041bc5ad16SMarcel Holtmann * However check that the packet type is valid. 16051bc5ad16SMarcel Holtmann */ 1606d79f34e3SMarcel Holtmann if (hci_skb_pkt_type(skb) != HCI_COMMAND_PKT && 1607d79f34e3SMarcel Holtmann hci_skb_pkt_type(skb) != HCI_ACLDATA_PKT && 1608d79f34e3SMarcel Holtmann hci_skb_pkt_type(skb) != HCI_SCODATA_PKT) { 16091bc5ad16SMarcel Holtmann err = -EINVAL; 16101bc5ad16SMarcel Holtmann goto drop; 16111bc5ad16SMarcel Holtmann } 16121bc5ad16SMarcel Holtmann 16131bc5ad16SMarcel Holtmann skb_queue_tail(&hdev->raw_q, skb); 16141bc5ad16SMarcel Holtmann queue_work(hdev->workqueue, &hdev->tx_work); 1615d79f34e3SMarcel Holtmann } else if (hci_skb_pkt_type(skb) == HCI_COMMAND_PKT) { 161683985319SHarvey Harrison u16 opcode = get_unaligned_le16(skb->data); 16171da177e4SLinus Torvalds u16 ogf = hci_opcode_ogf(opcode); 16181da177e4SLinus Torvalds u16 ocf = hci_opcode_ocf(opcode); 16191da177e4SLinus Torvalds 16201da177e4SLinus Torvalds if (((ogf > HCI_SFLT_MAX_OGF) || 16213bb3c755SGustavo Padovan !hci_test_bit(ocf & HCI_FLT_OCF_BITS, 16223bb3c755SGustavo Padovan &hci_sec_filter.ocf_mask[ogf])) && 16231da177e4SLinus Torvalds !capable(CAP_NET_RAW)) { 16241da177e4SLinus Torvalds err = -EPERM; 16251da177e4SLinus Torvalds goto drop; 16261da177e4SLinus Torvalds } 16271da177e4SLinus Torvalds 16281982162bSMarcel Holtmann /* Since the opcode has already been extracted here, store 16291982162bSMarcel Holtmann * a copy of the value for later use by the drivers. 16301982162bSMarcel Holtmann */ 16311982162bSMarcel Holtmann hci_skb_opcode(skb) = opcode; 16321982162bSMarcel Holtmann 1633fee746b0SMarcel Holtmann if (ogf == 0x3f) { 16341da177e4SLinus Torvalds skb_queue_tail(&hdev->raw_q, skb); 16353eff45eaSGustavo F. Padovan queue_work(hdev->workqueue, &hdev->tx_work); 16361da177e4SLinus Torvalds } else { 163749c922bbSStephen Hemminger /* Stand-alone HCI commands must be flagged as 163811714b3dSJohan Hedberg * single-command requests. 163911714b3dSJohan Hedberg */ 164044d27137SJohan Hedberg bt_cb(skb)->hci.req_flags |= HCI_REQ_START; 164111714b3dSJohan Hedberg 16421da177e4SLinus Torvalds skb_queue_tail(&hdev->cmd_q, skb); 1643c347b765SGustavo F. Padovan queue_work(hdev->workqueue, &hdev->cmd_work); 16441da177e4SLinus Torvalds } 16451da177e4SLinus Torvalds } else { 16461da177e4SLinus Torvalds if (!capable(CAP_NET_RAW)) { 16471da177e4SLinus Torvalds err = -EPERM; 16481da177e4SLinus Torvalds goto drop; 16491da177e4SLinus Torvalds } 16501da177e4SLinus Torvalds 1651d79f34e3SMarcel Holtmann if (hci_skb_pkt_type(skb) != HCI_ACLDATA_PKT && 1652d79f34e3SMarcel Holtmann hci_skb_pkt_type(skb) != HCI_SCODATA_PKT) { 1653bb77543eSMarcel Holtmann err = -EINVAL; 1654bb77543eSMarcel Holtmann goto drop; 1655bb77543eSMarcel Holtmann } 1656bb77543eSMarcel Holtmann 16571da177e4SLinus Torvalds skb_queue_tail(&hdev->raw_q, skb); 16583eff45eaSGustavo F. Padovan queue_work(hdev->workqueue, &hdev->tx_work); 16591da177e4SLinus Torvalds } 16601da177e4SLinus Torvalds 16611da177e4SLinus Torvalds err = len; 16621da177e4SLinus Torvalds 16631da177e4SLinus Torvalds done: 16641da177e4SLinus Torvalds release_sock(sk); 16651da177e4SLinus Torvalds return err; 16661da177e4SLinus Torvalds 16671da177e4SLinus Torvalds drop: 16681da177e4SLinus Torvalds kfree_skb(skb); 16691da177e4SLinus Torvalds goto done; 16701da177e4SLinus Torvalds } 16711da177e4SLinus Torvalds 16728fc9ced3SGustavo Padovan static int hci_sock_setsockopt(struct socket *sock, int level, int optname, 16738fc9ced3SGustavo Padovan char __user *optval, unsigned int len) 16741da177e4SLinus Torvalds { 16751da177e4SLinus Torvalds struct hci_ufilter uf = { .opcode = 0 }; 16761da177e4SLinus Torvalds struct sock *sk = sock->sk; 16771da177e4SLinus Torvalds int err = 0, opt = 0; 16781da177e4SLinus Torvalds 16791da177e4SLinus Torvalds BT_DBG("sk %p, opt %d", sk, optname); 16801da177e4SLinus Torvalds 168147b0f573SMarcel Holtmann if (level != SOL_HCI) 168247b0f573SMarcel Holtmann return -ENOPROTOOPT; 168347b0f573SMarcel Holtmann 16841da177e4SLinus Torvalds lock_sock(sk); 16851da177e4SLinus Torvalds 16862f39cdb7SMarcel Holtmann if (hci_pi(sk)->channel != HCI_CHANNEL_RAW) { 1687c2371e80SMarcel Holtmann err = -EBADFD; 16882f39cdb7SMarcel Holtmann goto done; 16892f39cdb7SMarcel Holtmann } 16902f39cdb7SMarcel Holtmann 16911da177e4SLinus Torvalds switch (optname) { 16921da177e4SLinus Torvalds case HCI_DATA_DIR: 16931da177e4SLinus Torvalds if (get_user(opt, (int __user *)optval)) { 16941da177e4SLinus Torvalds err = -EFAULT; 16951da177e4SLinus Torvalds break; 16961da177e4SLinus Torvalds } 16971da177e4SLinus Torvalds 16981da177e4SLinus Torvalds if (opt) 16991da177e4SLinus Torvalds hci_pi(sk)->cmsg_mask |= HCI_CMSG_DIR; 17001da177e4SLinus Torvalds else 17011da177e4SLinus Torvalds hci_pi(sk)->cmsg_mask &= ~HCI_CMSG_DIR; 17021da177e4SLinus Torvalds break; 17031da177e4SLinus Torvalds 17041da177e4SLinus Torvalds case HCI_TIME_STAMP: 17051da177e4SLinus Torvalds if (get_user(opt, (int __user *)optval)) { 17061da177e4SLinus Torvalds err = -EFAULT; 17071da177e4SLinus Torvalds break; 17081da177e4SLinus Torvalds } 17091da177e4SLinus Torvalds 17101da177e4SLinus Torvalds if (opt) 17111da177e4SLinus Torvalds hci_pi(sk)->cmsg_mask |= HCI_CMSG_TSTAMP; 17121da177e4SLinus Torvalds else 17131da177e4SLinus Torvalds hci_pi(sk)->cmsg_mask &= ~HCI_CMSG_TSTAMP; 17141da177e4SLinus Torvalds break; 17151da177e4SLinus Torvalds 17161da177e4SLinus Torvalds case HCI_FILTER: 17170878b666SMarcel Holtmann { 17180878b666SMarcel Holtmann struct hci_filter *f = &hci_pi(sk)->filter; 17190878b666SMarcel Holtmann 17200878b666SMarcel Holtmann uf.type_mask = f->type_mask; 17210878b666SMarcel Holtmann uf.opcode = f->opcode; 17220878b666SMarcel Holtmann uf.event_mask[0] = *((u32 *) f->event_mask + 0); 17230878b666SMarcel Holtmann uf.event_mask[1] = *((u32 *) f->event_mask + 1); 17240878b666SMarcel Holtmann } 17250878b666SMarcel Holtmann 17261da177e4SLinus Torvalds len = min_t(unsigned int, len, sizeof(uf)); 17271da177e4SLinus Torvalds if (copy_from_user(&uf, optval, len)) { 17281da177e4SLinus Torvalds err = -EFAULT; 17291da177e4SLinus Torvalds break; 17301da177e4SLinus Torvalds } 17311da177e4SLinus Torvalds 17321da177e4SLinus Torvalds if (!capable(CAP_NET_RAW)) { 17331da177e4SLinus Torvalds uf.type_mask &= hci_sec_filter.type_mask; 17341da177e4SLinus Torvalds uf.event_mask[0] &= *((u32 *) hci_sec_filter.event_mask + 0); 17351da177e4SLinus Torvalds uf.event_mask[1] &= *((u32 *) hci_sec_filter.event_mask + 1); 17361da177e4SLinus Torvalds } 17371da177e4SLinus Torvalds 17381da177e4SLinus Torvalds { 17391da177e4SLinus Torvalds struct hci_filter *f = &hci_pi(sk)->filter; 17401da177e4SLinus Torvalds 17411da177e4SLinus Torvalds f->type_mask = uf.type_mask; 17421da177e4SLinus Torvalds f->opcode = uf.opcode; 17431da177e4SLinus Torvalds *((u32 *) f->event_mask + 0) = uf.event_mask[0]; 17441da177e4SLinus Torvalds *((u32 *) f->event_mask + 1) = uf.event_mask[1]; 17451da177e4SLinus Torvalds } 17461da177e4SLinus Torvalds break; 17471da177e4SLinus Torvalds 17481da177e4SLinus Torvalds default: 17491da177e4SLinus Torvalds err = -ENOPROTOOPT; 17501da177e4SLinus Torvalds break; 17511da177e4SLinus Torvalds } 17521da177e4SLinus Torvalds 17532f39cdb7SMarcel Holtmann done: 17541da177e4SLinus Torvalds release_sock(sk); 17551da177e4SLinus Torvalds return err; 17561da177e4SLinus Torvalds } 17571da177e4SLinus Torvalds 17588fc9ced3SGustavo Padovan static int hci_sock_getsockopt(struct socket *sock, int level, int optname, 17598fc9ced3SGustavo Padovan char __user *optval, int __user *optlen) 17601da177e4SLinus Torvalds { 17611da177e4SLinus Torvalds struct hci_ufilter uf; 17621da177e4SLinus Torvalds struct sock *sk = sock->sk; 1763cedc5469SMarcel Holtmann int len, opt, err = 0; 1764cedc5469SMarcel Holtmann 1765cedc5469SMarcel Holtmann BT_DBG("sk %p, opt %d", sk, optname); 17661da177e4SLinus Torvalds 176747b0f573SMarcel Holtmann if (level != SOL_HCI) 176847b0f573SMarcel Holtmann return -ENOPROTOOPT; 176947b0f573SMarcel Holtmann 17701da177e4SLinus Torvalds if (get_user(len, optlen)) 17711da177e4SLinus Torvalds return -EFAULT; 17721da177e4SLinus Torvalds 1773cedc5469SMarcel Holtmann lock_sock(sk); 1774cedc5469SMarcel Holtmann 1775cedc5469SMarcel Holtmann if (hci_pi(sk)->channel != HCI_CHANNEL_RAW) { 1776c2371e80SMarcel Holtmann err = -EBADFD; 1777cedc5469SMarcel Holtmann goto done; 1778cedc5469SMarcel Holtmann } 1779cedc5469SMarcel Holtmann 17801da177e4SLinus Torvalds switch (optname) { 17811da177e4SLinus Torvalds case HCI_DATA_DIR: 17821da177e4SLinus Torvalds if (hci_pi(sk)->cmsg_mask & HCI_CMSG_DIR) 17831da177e4SLinus Torvalds opt = 1; 17841da177e4SLinus Torvalds else 17851da177e4SLinus Torvalds opt = 0; 17861da177e4SLinus Torvalds 17871da177e4SLinus Torvalds if (put_user(opt, optval)) 1788cedc5469SMarcel Holtmann err = -EFAULT; 17891da177e4SLinus Torvalds break; 17901da177e4SLinus Torvalds 17911da177e4SLinus Torvalds case HCI_TIME_STAMP: 17921da177e4SLinus Torvalds if (hci_pi(sk)->cmsg_mask & HCI_CMSG_TSTAMP) 17931da177e4SLinus Torvalds opt = 1; 17941da177e4SLinus Torvalds else 17951da177e4SLinus Torvalds opt = 0; 17961da177e4SLinus Torvalds 17971da177e4SLinus Torvalds if (put_user(opt, optval)) 1798cedc5469SMarcel Holtmann err = -EFAULT; 17991da177e4SLinus Torvalds break; 18001da177e4SLinus Torvalds 18011da177e4SLinus Torvalds case HCI_FILTER: 18021da177e4SLinus Torvalds { 18031da177e4SLinus Torvalds struct hci_filter *f = &hci_pi(sk)->filter; 18041da177e4SLinus Torvalds 1805e15ca9a0SMathias Krause memset(&uf, 0, sizeof(uf)); 18061da177e4SLinus Torvalds uf.type_mask = f->type_mask; 18071da177e4SLinus Torvalds uf.opcode = f->opcode; 18081da177e4SLinus Torvalds uf.event_mask[0] = *((u32 *) f->event_mask + 0); 18091da177e4SLinus Torvalds uf.event_mask[1] = *((u32 *) f->event_mask + 1); 18101da177e4SLinus Torvalds } 18111da177e4SLinus Torvalds 18121da177e4SLinus Torvalds len = min_t(unsigned int, len, sizeof(uf)); 18131da177e4SLinus Torvalds if (copy_to_user(optval, &uf, len)) 1814cedc5469SMarcel Holtmann err = -EFAULT; 18151da177e4SLinus Torvalds break; 18161da177e4SLinus Torvalds 18171da177e4SLinus Torvalds default: 1818cedc5469SMarcel Holtmann err = -ENOPROTOOPT; 18191da177e4SLinus Torvalds break; 18201da177e4SLinus Torvalds } 18211da177e4SLinus Torvalds 1822cedc5469SMarcel Holtmann done: 1823cedc5469SMarcel Holtmann release_sock(sk); 1824cedc5469SMarcel Holtmann return err; 18251da177e4SLinus Torvalds } 18261da177e4SLinus Torvalds 182790ddc4f0SEric Dumazet static const struct proto_ops hci_sock_ops = { 18281da177e4SLinus Torvalds .family = PF_BLUETOOTH, 18291da177e4SLinus Torvalds .owner = THIS_MODULE, 18301da177e4SLinus Torvalds .release = hci_sock_release, 18311da177e4SLinus Torvalds .bind = hci_sock_bind, 18321da177e4SLinus Torvalds .getname = hci_sock_getname, 18331da177e4SLinus Torvalds .sendmsg = hci_sock_sendmsg, 18341da177e4SLinus Torvalds .recvmsg = hci_sock_recvmsg, 18351da177e4SLinus Torvalds .ioctl = hci_sock_ioctl, 18361da177e4SLinus Torvalds .poll = datagram_poll, 18371da177e4SLinus Torvalds .listen = sock_no_listen, 18381da177e4SLinus Torvalds .shutdown = sock_no_shutdown, 18391da177e4SLinus Torvalds .setsockopt = hci_sock_setsockopt, 18401da177e4SLinus Torvalds .getsockopt = hci_sock_getsockopt, 18411da177e4SLinus Torvalds .connect = sock_no_connect, 18421da177e4SLinus Torvalds .socketpair = sock_no_socketpair, 18431da177e4SLinus Torvalds .accept = sock_no_accept, 18441da177e4SLinus Torvalds .mmap = sock_no_mmap 18451da177e4SLinus Torvalds }; 18461da177e4SLinus Torvalds 18471da177e4SLinus Torvalds static struct proto hci_sk_proto = { 18481da177e4SLinus Torvalds .name = "HCI", 18491da177e4SLinus Torvalds .owner = THIS_MODULE, 18501da177e4SLinus Torvalds .obj_size = sizeof(struct hci_pinfo) 18511da177e4SLinus Torvalds }; 18521da177e4SLinus Torvalds 18533f378b68SEric Paris static int hci_sock_create(struct net *net, struct socket *sock, int protocol, 18543f378b68SEric Paris int kern) 18551da177e4SLinus Torvalds { 18561da177e4SLinus Torvalds struct sock *sk; 18571da177e4SLinus Torvalds 18581da177e4SLinus Torvalds BT_DBG("sock %p", sock); 18591da177e4SLinus Torvalds 18601da177e4SLinus Torvalds if (sock->type != SOCK_RAW) 18611da177e4SLinus Torvalds return -ESOCKTNOSUPPORT; 18621da177e4SLinus Torvalds 18631da177e4SLinus Torvalds sock->ops = &hci_sock_ops; 18641da177e4SLinus Torvalds 186511aa9c28SEric W. Biederman sk = sk_alloc(net, PF_BLUETOOTH, GFP_ATOMIC, &hci_sk_proto, kern); 18661da177e4SLinus Torvalds if (!sk) 18671da177e4SLinus Torvalds return -ENOMEM; 18681da177e4SLinus Torvalds 18691da177e4SLinus Torvalds sock_init_data(sock, sk); 18701da177e4SLinus Torvalds 18711da177e4SLinus Torvalds sock_reset_flag(sk, SOCK_ZAPPED); 18721da177e4SLinus Torvalds 18731da177e4SLinus Torvalds sk->sk_protocol = protocol; 18741da177e4SLinus Torvalds 18751da177e4SLinus Torvalds sock->state = SS_UNCONNECTED; 18761da177e4SLinus Torvalds sk->sk_state = BT_OPEN; 18771da177e4SLinus Torvalds 18781da177e4SLinus Torvalds bt_sock_link(&hci_sk_list, sk); 18791da177e4SLinus Torvalds return 0; 18801da177e4SLinus Torvalds } 18811da177e4SLinus Torvalds 1882ec1b4cf7SStephen Hemminger static const struct net_proto_family hci_sock_family_ops = { 18831da177e4SLinus Torvalds .family = PF_BLUETOOTH, 18841da177e4SLinus Torvalds .owner = THIS_MODULE, 18851da177e4SLinus Torvalds .create = hci_sock_create, 18861da177e4SLinus Torvalds }; 18871da177e4SLinus Torvalds 18881da177e4SLinus Torvalds int __init hci_sock_init(void) 18891da177e4SLinus Torvalds { 18901da177e4SLinus Torvalds int err; 18911da177e4SLinus Torvalds 1892b0a8e282SMarcel Holtmann BUILD_BUG_ON(sizeof(struct sockaddr_hci) > sizeof(struct sockaddr)); 1893b0a8e282SMarcel Holtmann 18941da177e4SLinus Torvalds err = proto_register(&hci_sk_proto, 0); 18951da177e4SLinus Torvalds if (err < 0) 18961da177e4SLinus Torvalds return err; 18971da177e4SLinus Torvalds 18981da177e4SLinus Torvalds err = bt_sock_register(BTPROTO_HCI, &hci_sock_family_ops); 1899f7c86637SMasatake YAMATO if (err < 0) { 1900f7c86637SMasatake YAMATO BT_ERR("HCI socket registration failed"); 19011da177e4SLinus Torvalds goto error; 1902f7c86637SMasatake YAMATO } 1903f7c86637SMasatake YAMATO 1904b0316615SAl Viro err = bt_procfs_init(&init_net, "hci", &hci_sk_list, NULL); 1905f7c86637SMasatake YAMATO if (err < 0) { 1906f7c86637SMasatake YAMATO BT_ERR("Failed to create HCI proc file"); 1907f7c86637SMasatake YAMATO bt_sock_unregister(BTPROTO_HCI); 1908f7c86637SMasatake YAMATO goto error; 1909f7c86637SMasatake YAMATO } 19101da177e4SLinus Torvalds 19111da177e4SLinus Torvalds BT_INFO("HCI socket layer initialized"); 19121da177e4SLinus Torvalds 19131da177e4SLinus Torvalds return 0; 19141da177e4SLinus Torvalds 19151da177e4SLinus Torvalds error: 19161da177e4SLinus Torvalds proto_unregister(&hci_sk_proto); 19171da177e4SLinus Torvalds return err; 19181da177e4SLinus Torvalds } 19191da177e4SLinus Torvalds 1920b7440a14SAnand Gadiyar void hci_sock_cleanup(void) 19211da177e4SLinus Torvalds { 1922f7c86637SMasatake YAMATO bt_procfs_cleanup(&init_net, "hci"); 19235e9d7f86SDavid Herrmann bt_sock_unregister(BTPROTO_HCI); 19241da177e4SLinus Torvalds proto_unregister(&hci_sk_proto); 19251da177e4SLinus Torvalds } 1926