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; 482d0bef1d2SMarcel Holtmann u16 format; 483249fa169SMarcel Holtmann u8 ver[3]; 484249fa169SMarcel Holtmann u32 flags; 485249fa169SMarcel Holtmann 4860ef2c42fSMarcel Holtmann /* No message needed when cookie is not present */ 4870ef2c42fSMarcel Holtmann if (!hci_pi(sk)->cookie) 4880ef2c42fSMarcel Holtmann return NULL; 4890ef2c42fSMarcel Holtmann 490d0bef1d2SMarcel Holtmann switch (hci_pi(sk)->channel) { 491*f81f5b2dSMarcel Holtmann case HCI_CHANNEL_RAW: 492*f81f5b2dSMarcel Holtmann format = 0x0000; 493*f81f5b2dSMarcel Holtmann ver[0] = BT_SUBSYS_VERSION; 494*f81f5b2dSMarcel Holtmann put_unaligned_le16(BT_SUBSYS_REVISION, ver + 1); 495*f81f5b2dSMarcel Holtmann break; 496d0bef1d2SMarcel Holtmann case HCI_CHANNEL_CONTROL: 497d0bef1d2SMarcel Holtmann format = 0x0002; 498d0bef1d2SMarcel Holtmann mgmt_fill_version_info(ver); 499d0bef1d2SMarcel Holtmann break; 500d0bef1d2SMarcel Holtmann default: 501d0bef1d2SMarcel Holtmann /* No message for unsupported format */ 502d0bef1d2SMarcel Holtmann return NULL; 503d0bef1d2SMarcel Holtmann } 504d0bef1d2SMarcel Holtmann 505249fa169SMarcel Holtmann skb = bt_skb_alloc(14 + TASK_COMM_LEN , GFP_ATOMIC); 506249fa169SMarcel Holtmann if (!skb) 507249fa169SMarcel Holtmann return NULL; 508249fa169SMarcel Holtmann 509249fa169SMarcel Holtmann flags = hci_sock_test_flag(sk, HCI_SOCK_TRUSTED) ? 0x1 : 0x0; 510249fa169SMarcel Holtmann 511249fa169SMarcel Holtmann put_unaligned_le32(hci_pi(sk)->cookie, skb_put(skb, 4)); 512249fa169SMarcel Holtmann put_unaligned_le16(format, skb_put(skb, 2)); 513249fa169SMarcel Holtmann memcpy(skb_put(skb, sizeof(ver)), ver, sizeof(ver)); 514249fa169SMarcel Holtmann put_unaligned_le32(flags, skb_put(skb, 4)); 515249fa169SMarcel Holtmann *skb_put(skb, 1) = TASK_COMM_LEN; 516249fa169SMarcel Holtmann memcpy(skb_put(skb, TASK_COMM_LEN), hci_pi(sk)->comm, TASK_COMM_LEN); 517249fa169SMarcel Holtmann 518249fa169SMarcel Holtmann __net_timestamp(skb); 519249fa169SMarcel Holtmann 520249fa169SMarcel Holtmann hdr = (void *)skb_push(skb, HCI_MON_HDR_SIZE); 521249fa169SMarcel Holtmann hdr->opcode = cpu_to_le16(HCI_MON_CTRL_OPEN); 5220ef2c42fSMarcel Holtmann if (hci_pi(sk)->hdev) 5230ef2c42fSMarcel Holtmann hdr->index = cpu_to_le16(hci_pi(sk)->hdev->id); 5240ef2c42fSMarcel Holtmann else 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 531249fa169SMarcel Holtmann static struct sk_buff *create_monitor_ctrl_close(struct sock *sk) 532249fa169SMarcel Holtmann { 533249fa169SMarcel Holtmann struct hci_mon_hdr *hdr; 534249fa169SMarcel Holtmann struct sk_buff *skb; 535249fa169SMarcel Holtmann 5360ef2c42fSMarcel Holtmann /* No message needed when cookie is not present */ 5370ef2c42fSMarcel Holtmann if (!hci_pi(sk)->cookie) 5380ef2c42fSMarcel Holtmann return NULL; 5390ef2c42fSMarcel Holtmann 540d0bef1d2SMarcel Holtmann switch (hci_pi(sk)->channel) { 541*f81f5b2dSMarcel Holtmann case HCI_CHANNEL_RAW: 542d0bef1d2SMarcel Holtmann case HCI_CHANNEL_CONTROL: 543d0bef1d2SMarcel Holtmann break; 544d0bef1d2SMarcel Holtmann default: 545d0bef1d2SMarcel Holtmann /* No message for unsupported format */ 546d0bef1d2SMarcel Holtmann return NULL; 547d0bef1d2SMarcel Holtmann } 548d0bef1d2SMarcel Holtmann 549249fa169SMarcel Holtmann skb = bt_skb_alloc(4, GFP_ATOMIC); 550249fa169SMarcel Holtmann if (!skb) 551249fa169SMarcel Holtmann return NULL; 552249fa169SMarcel Holtmann 553249fa169SMarcel Holtmann put_unaligned_le32(hci_pi(sk)->cookie, skb_put(skb, 4)); 554249fa169SMarcel Holtmann 555249fa169SMarcel Holtmann __net_timestamp(skb); 556249fa169SMarcel Holtmann 557249fa169SMarcel Holtmann hdr = (void *)skb_push(skb, HCI_MON_HDR_SIZE); 558249fa169SMarcel Holtmann hdr->opcode = cpu_to_le16(HCI_MON_CTRL_CLOSE); 5590ef2c42fSMarcel Holtmann if (hci_pi(sk)->hdev) 5600ef2c42fSMarcel Holtmann hdr->index = cpu_to_le16(hci_pi(sk)->hdev->id); 5610ef2c42fSMarcel Holtmann else 562249fa169SMarcel Holtmann hdr->index = cpu_to_le16(HCI_DEV_NONE); 563249fa169SMarcel Holtmann hdr->len = cpu_to_le16(skb->len - HCI_MON_HDR_SIZE); 564249fa169SMarcel Holtmann 565249fa169SMarcel Holtmann return skb; 566249fa169SMarcel Holtmann } 567249fa169SMarcel Holtmann 56838ceaa00SMarcel Holtmann static struct sk_buff *create_monitor_ctrl_command(struct sock *sk, u16 index, 56938ceaa00SMarcel Holtmann u16 opcode, u16 len, 57038ceaa00SMarcel Holtmann const void *buf) 57138ceaa00SMarcel Holtmann { 57238ceaa00SMarcel Holtmann struct hci_mon_hdr *hdr; 57338ceaa00SMarcel Holtmann struct sk_buff *skb; 57438ceaa00SMarcel Holtmann 57538ceaa00SMarcel Holtmann skb = bt_skb_alloc(6 + len, GFP_ATOMIC); 57638ceaa00SMarcel Holtmann if (!skb) 57738ceaa00SMarcel Holtmann return NULL; 57838ceaa00SMarcel Holtmann 57938ceaa00SMarcel Holtmann put_unaligned_le32(hci_pi(sk)->cookie, skb_put(skb, 4)); 58038ceaa00SMarcel Holtmann put_unaligned_le16(opcode, skb_put(skb, 2)); 58138ceaa00SMarcel Holtmann 58238ceaa00SMarcel Holtmann if (buf) 58338ceaa00SMarcel Holtmann memcpy(skb_put(skb, len), buf, len); 58438ceaa00SMarcel Holtmann 58538ceaa00SMarcel Holtmann __net_timestamp(skb); 58638ceaa00SMarcel Holtmann 58738ceaa00SMarcel Holtmann hdr = (void *)skb_push(skb, HCI_MON_HDR_SIZE); 58838ceaa00SMarcel Holtmann hdr->opcode = cpu_to_le16(HCI_MON_CTRL_COMMAND); 58938ceaa00SMarcel Holtmann hdr->index = cpu_to_le16(index); 59038ceaa00SMarcel Holtmann hdr->len = cpu_to_le16(skb->len - HCI_MON_HDR_SIZE); 59138ceaa00SMarcel Holtmann 59238ceaa00SMarcel Holtmann return skb; 59338ceaa00SMarcel Holtmann } 59438ceaa00SMarcel Holtmann 595787b306cSJohannes Berg static void __printf(2, 3) 596787b306cSJohannes Berg send_monitor_note(struct sock *sk, const char *fmt, ...) 597dd31506dSMarcel Holtmann { 598787b306cSJohannes Berg size_t len; 599dd31506dSMarcel Holtmann struct hci_mon_hdr *hdr; 600dd31506dSMarcel Holtmann struct sk_buff *skb; 601787b306cSJohannes Berg va_list args; 602787b306cSJohannes Berg 603787b306cSJohannes Berg va_start(args, fmt); 604787b306cSJohannes Berg len = vsnprintf(NULL, 0, fmt, args); 605787b306cSJohannes Berg va_end(args); 606dd31506dSMarcel Holtmann 607dd31506dSMarcel Holtmann skb = bt_skb_alloc(len + 1, GFP_ATOMIC); 608dd31506dSMarcel Holtmann if (!skb) 609dd31506dSMarcel Holtmann return; 610dd31506dSMarcel Holtmann 611787b306cSJohannes Berg va_start(args, fmt); 612787b306cSJohannes Berg vsprintf(skb_put(skb, len), fmt, args); 613787b306cSJohannes Berg *skb_put(skb, 1) = 0; 614787b306cSJohannes Berg va_end(args); 615dd31506dSMarcel Holtmann 616dd31506dSMarcel Holtmann __net_timestamp(skb); 617dd31506dSMarcel Holtmann 618dd31506dSMarcel Holtmann hdr = (void *)skb_push(skb, HCI_MON_HDR_SIZE); 619dd31506dSMarcel Holtmann hdr->opcode = cpu_to_le16(HCI_MON_SYSTEM_NOTE); 620dd31506dSMarcel Holtmann hdr->index = cpu_to_le16(HCI_DEV_NONE); 621dd31506dSMarcel Holtmann hdr->len = cpu_to_le16(skb->len - HCI_MON_HDR_SIZE); 622dd31506dSMarcel Holtmann 623dd31506dSMarcel Holtmann if (sock_queue_rcv_skb(sk, skb)) 624dd31506dSMarcel Holtmann kfree_skb(skb); 625dd31506dSMarcel Holtmann } 626dd31506dSMarcel Holtmann 627cd82e61cSMarcel Holtmann static void send_monitor_replay(struct sock *sk) 628cd82e61cSMarcel Holtmann { 629cd82e61cSMarcel Holtmann struct hci_dev *hdev; 630cd82e61cSMarcel Holtmann 631cd82e61cSMarcel Holtmann read_lock(&hci_dev_list_lock); 632cd82e61cSMarcel Holtmann 633cd82e61cSMarcel Holtmann list_for_each_entry(hdev, &hci_dev_list, list) { 634cd82e61cSMarcel Holtmann struct sk_buff *skb; 635cd82e61cSMarcel Holtmann 636cd82e61cSMarcel Holtmann skb = create_monitor_event(hdev, HCI_DEV_REG); 637cd82e61cSMarcel Holtmann if (!skb) 638cd82e61cSMarcel Holtmann continue; 639cd82e61cSMarcel Holtmann 640cd82e61cSMarcel Holtmann if (sock_queue_rcv_skb(sk, skb)) 641cd82e61cSMarcel Holtmann kfree_skb(skb); 64222db3cbcSMarcel Holtmann 64322db3cbcSMarcel Holtmann if (!test_bit(HCI_RUNNING, &hdev->flags)) 64422db3cbcSMarcel Holtmann continue; 64522db3cbcSMarcel Holtmann 64622db3cbcSMarcel Holtmann skb = create_monitor_event(hdev, HCI_DEV_OPEN); 64722db3cbcSMarcel Holtmann if (!skb) 64822db3cbcSMarcel Holtmann continue; 64922db3cbcSMarcel Holtmann 65022db3cbcSMarcel Holtmann if (sock_queue_rcv_skb(sk, skb)) 65122db3cbcSMarcel Holtmann kfree_skb(skb); 6526c566dd5SMarcel Holtmann 653e131d74aSMarcel Holtmann if (test_bit(HCI_UP, &hdev->flags)) 6546c566dd5SMarcel Holtmann skb = create_monitor_event(hdev, HCI_DEV_UP); 655e131d74aSMarcel Holtmann else if (hci_dev_test_flag(hdev, HCI_SETUP)) 656e131d74aSMarcel Holtmann skb = create_monitor_event(hdev, HCI_DEV_SETUP); 657e131d74aSMarcel Holtmann else 658e131d74aSMarcel Holtmann skb = NULL; 6596c566dd5SMarcel Holtmann 660e131d74aSMarcel Holtmann if (skb) { 6616c566dd5SMarcel Holtmann if (sock_queue_rcv_skb(sk, skb)) 6626c566dd5SMarcel Holtmann kfree_skb(skb); 663cd82e61cSMarcel Holtmann } 664e131d74aSMarcel Holtmann } 665cd82e61cSMarcel Holtmann 666cd82e61cSMarcel Holtmann read_unlock(&hci_dev_list_lock); 667cd82e61cSMarcel Holtmann } 668cd82e61cSMarcel Holtmann 669249fa169SMarcel Holtmann static void send_monitor_control_replay(struct sock *mon_sk) 670249fa169SMarcel Holtmann { 671249fa169SMarcel Holtmann struct sock *sk; 672249fa169SMarcel Holtmann 673249fa169SMarcel Holtmann read_lock(&hci_sk_list.lock); 674249fa169SMarcel Holtmann 675249fa169SMarcel Holtmann sk_for_each(sk, &hci_sk_list.head) { 676249fa169SMarcel Holtmann struct sk_buff *skb; 677249fa169SMarcel Holtmann 678249fa169SMarcel Holtmann skb = create_monitor_ctrl_open(sk); 679249fa169SMarcel Holtmann if (!skb) 680249fa169SMarcel Holtmann continue; 681249fa169SMarcel Holtmann 682249fa169SMarcel Holtmann if (sock_queue_rcv_skb(mon_sk, skb)) 683249fa169SMarcel Holtmann kfree_skb(skb); 684249fa169SMarcel Holtmann } 685249fa169SMarcel Holtmann 686249fa169SMarcel Holtmann read_unlock(&hci_sk_list.lock); 687249fa169SMarcel Holtmann } 688249fa169SMarcel Holtmann 689040030efSMarcel Holtmann /* Generate internal stack event */ 690040030efSMarcel Holtmann static void hci_si_event(struct hci_dev *hdev, int type, int dlen, void *data) 691040030efSMarcel Holtmann { 692040030efSMarcel Holtmann struct hci_event_hdr *hdr; 693040030efSMarcel Holtmann struct hci_ev_stack_internal *ev; 694040030efSMarcel Holtmann struct sk_buff *skb; 695040030efSMarcel Holtmann 696040030efSMarcel Holtmann skb = bt_skb_alloc(HCI_EVENT_HDR_SIZE + sizeof(*ev) + dlen, GFP_ATOMIC); 697040030efSMarcel Holtmann if (!skb) 698040030efSMarcel Holtmann return; 699040030efSMarcel Holtmann 700040030efSMarcel Holtmann hdr = (void *)skb_put(skb, HCI_EVENT_HDR_SIZE); 701040030efSMarcel Holtmann hdr->evt = HCI_EV_STACK_INTERNAL; 702040030efSMarcel Holtmann hdr->plen = sizeof(*ev) + dlen; 703040030efSMarcel Holtmann 704040030efSMarcel Holtmann ev = (void *)skb_put(skb, sizeof(*ev) + dlen); 705040030efSMarcel Holtmann ev->type = type; 706040030efSMarcel Holtmann memcpy(ev->data, data, dlen); 707040030efSMarcel Holtmann 708040030efSMarcel Holtmann bt_cb(skb)->incoming = 1; 709040030efSMarcel Holtmann __net_timestamp(skb); 710040030efSMarcel Holtmann 711d79f34e3SMarcel Holtmann hci_skb_pkt_type(skb) = HCI_EVENT_PKT; 712040030efSMarcel Holtmann hci_send_to_sock(hdev, skb); 713040030efSMarcel Holtmann kfree_skb(skb); 714040030efSMarcel Holtmann } 715040030efSMarcel Holtmann 716040030efSMarcel Holtmann void hci_sock_dev_event(struct hci_dev *hdev, int event) 717040030efSMarcel Holtmann { 718040030efSMarcel Holtmann BT_DBG("hdev %s event %d", hdev->name, event); 719040030efSMarcel Holtmann 720cd82e61cSMarcel Holtmann if (atomic_read(&monitor_promisc)) { 721cd82e61cSMarcel Holtmann struct sk_buff *skb; 722cd82e61cSMarcel Holtmann 723ed1b28a4SMarcel Holtmann /* Send event to monitor */ 724cd82e61cSMarcel Holtmann skb = create_monitor_event(hdev, event); 725cd82e61cSMarcel Holtmann if (skb) { 726c08b1a1dSMarcel Holtmann hci_send_to_channel(HCI_CHANNEL_MONITOR, skb, 727c08b1a1dSMarcel Holtmann HCI_SOCK_TRUSTED, NULL); 728cd82e61cSMarcel Holtmann kfree_skb(skb); 729cd82e61cSMarcel Holtmann } 730cd82e61cSMarcel Holtmann } 731cd82e61cSMarcel Holtmann 732ed1b28a4SMarcel Holtmann if (event <= HCI_DEV_DOWN) { 733ed1b28a4SMarcel Holtmann struct hci_ev_si_device ev; 734ed1b28a4SMarcel Holtmann 735040030efSMarcel Holtmann /* Send event to sockets */ 736040030efSMarcel Holtmann ev.event = event; 737040030efSMarcel Holtmann ev.dev_id = hdev->id; 738040030efSMarcel Holtmann hci_si_event(NULL, HCI_EV_SI_DEVICE, sizeof(ev), &ev); 739ed1b28a4SMarcel Holtmann } 740040030efSMarcel Holtmann 741040030efSMarcel Holtmann if (event == HCI_DEV_UNREG) { 742040030efSMarcel Holtmann struct sock *sk; 743040030efSMarcel Holtmann 744040030efSMarcel Holtmann /* Detach sockets from device */ 745040030efSMarcel Holtmann read_lock(&hci_sk_list.lock); 746b67bfe0dSSasha Levin sk_for_each(sk, &hci_sk_list.head) { 747040030efSMarcel Holtmann bh_lock_sock_nested(sk); 748040030efSMarcel Holtmann if (hci_pi(sk)->hdev == hdev) { 749040030efSMarcel Holtmann hci_pi(sk)->hdev = NULL; 750040030efSMarcel Holtmann sk->sk_err = EPIPE; 751040030efSMarcel Holtmann sk->sk_state = BT_OPEN; 752040030efSMarcel Holtmann sk->sk_state_change(sk); 753040030efSMarcel Holtmann 754040030efSMarcel Holtmann hci_dev_put(hdev); 755040030efSMarcel Holtmann } 756040030efSMarcel Holtmann bh_unlock_sock(sk); 757040030efSMarcel Holtmann } 758040030efSMarcel Holtmann read_unlock(&hci_sk_list.lock); 759040030efSMarcel Holtmann } 760040030efSMarcel Holtmann } 761040030efSMarcel Holtmann 762801c1e8dSJohan Hedberg static struct hci_mgmt_chan *__hci_mgmt_chan_find(unsigned short channel) 763801c1e8dSJohan Hedberg { 764801c1e8dSJohan Hedberg struct hci_mgmt_chan *c; 765801c1e8dSJohan Hedberg 766801c1e8dSJohan Hedberg list_for_each_entry(c, &mgmt_chan_list, list) { 767801c1e8dSJohan Hedberg if (c->channel == channel) 768801c1e8dSJohan Hedberg return c; 769801c1e8dSJohan Hedberg } 770801c1e8dSJohan Hedberg 771801c1e8dSJohan Hedberg return NULL; 772801c1e8dSJohan Hedberg } 773801c1e8dSJohan Hedberg 774801c1e8dSJohan Hedberg static struct hci_mgmt_chan *hci_mgmt_chan_find(unsigned short channel) 775801c1e8dSJohan Hedberg { 776801c1e8dSJohan Hedberg struct hci_mgmt_chan *c; 777801c1e8dSJohan Hedberg 778801c1e8dSJohan Hedberg mutex_lock(&mgmt_chan_list_lock); 779801c1e8dSJohan Hedberg c = __hci_mgmt_chan_find(channel); 780801c1e8dSJohan Hedberg mutex_unlock(&mgmt_chan_list_lock); 781801c1e8dSJohan Hedberg 782801c1e8dSJohan Hedberg return c; 783801c1e8dSJohan Hedberg } 784801c1e8dSJohan Hedberg 785801c1e8dSJohan Hedberg int hci_mgmt_chan_register(struct hci_mgmt_chan *c) 786801c1e8dSJohan Hedberg { 787801c1e8dSJohan Hedberg if (c->channel < HCI_CHANNEL_CONTROL) 788801c1e8dSJohan Hedberg return -EINVAL; 789801c1e8dSJohan Hedberg 790801c1e8dSJohan Hedberg mutex_lock(&mgmt_chan_list_lock); 791801c1e8dSJohan Hedberg if (__hci_mgmt_chan_find(c->channel)) { 792801c1e8dSJohan Hedberg mutex_unlock(&mgmt_chan_list_lock); 793801c1e8dSJohan Hedberg return -EALREADY; 794801c1e8dSJohan Hedberg } 795801c1e8dSJohan Hedberg 796801c1e8dSJohan Hedberg list_add_tail(&c->list, &mgmt_chan_list); 797801c1e8dSJohan Hedberg 798801c1e8dSJohan Hedberg mutex_unlock(&mgmt_chan_list_lock); 799801c1e8dSJohan Hedberg 800801c1e8dSJohan Hedberg return 0; 801801c1e8dSJohan Hedberg } 802801c1e8dSJohan Hedberg EXPORT_SYMBOL(hci_mgmt_chan_register); 803801c1e8dSJohan Hedberg 804801c1e8dSJohan Hedberg void hci_mgmt_chan_unregister(struct hci_mgmt_chan *c) 805801c1e8dSJohan Hedberg { 806801c1e8dSJohan Hedberg mutex_lock(&mgmt_chan_list_lock); 807801c1e8dSJohan Hedberg list_del(&c->list); 808801c1e8dSJohan Hedberg mutex_unlock(&mgmt_chan_list_lock); 809801c1e8dSJohan Hedberg } 810801c1e8dSJohan Hedberg EXPORT_SYMBOL(hci_mgmt_chan_unregister); 811801c1e8dSJohan Hedberg 8121da177e4SLinus Torvalds static int hci_sock_release(struct socket *sock) 8131da177e4SLinus Torvalds { 8141da177e4SLinus Torvalds struct sock *sk = sock->sk; 8157b005bd3SMarcel Holtmann struct hci_dev *hdev; 816249fa169SMarcel Holtmann struct sk_buff *skb; 8171da177e4SLinus Torvalds 8181da177e4SLinus Torvalds BT_DBG("sock %p sk %p", sock, sk); 8191da177e4SLinus Torvalds 8201da177e4SLinus Torvalds if (!sk) 8211da177e4SLinus Torvalds return 0; 8221da177e4SLinus Torvalds 8237b005bd3SMarcel Holtmann hdev = hci_pi(sk)->hdev; 8247b005bd3SMarcel Holtmann 82570ecce91SMarcel Holtmann switch (hci_pi(sk)->channel) { 82670ecce91SMarcel Holtmann case HCI_CHANNEL_MONITOR: 827cd82e61cSMarcel Holtmann atomic_dec(&monitor_promisc); 82870ecce91SMarcel Holtmann break; 829*f81f5b2dSMarcel Holtmann case HCI_CHANNEL_RAW: 83070ecce91SMarcel Holtmann case HCI_CHANNEL_CONTROL: 831249fa169SMarcel Holtmann /* Send event to monitor */ 832249fa169SMarcel Holtmann skb = create_monitor_ctrl_close(sk); 833249fa169SMarcel Holtmann if (skb) { 834249fa169SMarcel Holtmann hci_send_to_channel(HCI_CHANNEL_MONITOR, skb, 835249fa169SMarcel Holtmann HCI_SOCK_TRUSTED, NULL); 836249fa169SMarcel Holtmann kfree_skb(skb); 837249fa169SMarcel Holtmann } 838249fa169SMarcel Holtmann 839df1cb87aSMarcel Holtmann hci_sock_free_cookie(sk); 84070ecce91SMarcel Holtmann break; 84170ecce91SMarcel Holtmann } 842cd82e61cSMarcel Holtmann 8431da177e4SLinus Torvalds bt_sock_unlink(&hci_sk_list, sk); 8441da177e4SLinus Torvalds 8451da177e4SLinus Torvalds if (hdev) { 84623500189SMarcel Holtmann if (hci_pi(sk)->channel == HCI_CHANNEL_USER) { 8476b3cc1dbSSimon Fels /* When releasing an user channel exclusive access, 8486b3cc1dbSSimon Fels * call hci_dev_do_close directly instead of calling 8496b3cc1dbSSimon Fels * hci_dev_close to ensure the exclusive access will 8506b3cc1dbSSimon Fels * be released and the controller brought back down. 8516b3cc1dbSSimon Fels * 8526b3cc1dbSSimon Fels * The checking of HCI_AUTO_OFF is not needed in this 8536b3cc1dbSSimon Fels * case since it will have been cleared already when 8546b3cc1dbSSimon Fels * opening the user channel. 8556b3cc1dbSSimon Fels */ 8566b3cc1dbSSimon Fels hci_dev_do_close(hdev); 8579380f9eaSLoic Poulain hci_dev_clear_flag(hdev, HCI_USER_CHANNEL); 8589380f9eaSLoic Poulain mgmt_index_added(hdev); 85923500189SMarcel Holtmann } 86023500189SMarcel Holtmann 8611da177e4SLinus Torvalds atomic_dec(&hdev->promisc); 8621da177e4SLinus Torvalds hci_dev_put(hdev); 8631da177e4SLinus Torvalds } 8641da177e4SLinus Torvalds 8651da177e4SLinus Torvalds sock_orphan(sk); 8661da177e4SLinus Torvalds 8671da177e4SLinus Torvalds skb_queue_purge(&sk->sk_receive_queue); 8681da177e4SLinus Torvalds skb_queue_purge(&sk->sk_write_queue); 8691da177e4SLinus Torvalds 8701da177e4SLinus Torvalds sock_put(sk); 8711da177e4SLinus Torvalds return 0; 8721da177e4SLinus Torvalds } 8731da177e4SLinus Torvalds 874b2a66aadSAntti Julku static int hci_sock_blacklist_add(struct hci_dev *hdev, void __user *arg) 875f0358568SJohan Hedberg { 876f0358568SJohan Hedberg bdaddr_t bdaddr; 8775e762444SAntti Julku int err; 878f0358568SJohan Hedberg 879f0358568SJohan Hedberg if (copy_from_user(&bdaddr, arg, sizeof(bdaddr))) 880f0358568SJohan Hedberg return -EFAULT; 881f0358568SJohan Hedberg 88209fd0de5SGustavo F. Padovan hci_dev_lock(hdev); 8835e762444SAntti Julku 884dcc36c16SJohan Hedberg err = hci_bdaddr_list_add(&hdev->blacklist, &bdaddr, BDADDR_BREDR); 8855e762444SAntti Julku 88609fd0de5SGustavo F. Padovan hci_dev_unlock(hdev); 8875e762444SAntti Julku 8885e762444SAntti Julku return err; 889f0358568SJohan Hedberg } 890f0358568SJohan Hedberg 891b2a66aadSAntti Julku static int hci_sock_blacklist_del(struct hci_dev *hdev, void __user *arg) 892f0358568SJohan Hedberg { 893f0358568SJohan Hedberg bdaddr_t bdaddr; 8945e762444SAntti Julku int err; 895f0358568SJohan Hedberg 896f0358568SJohan Hedberg if (copy_from_user(&bdaddr, arg, sizeof(bdaddr))) 897f0358568SJohan Hedberg return -EFAULT; 898f0358568SJohan Hedberg 89909fd0de5SGustavo F. Padovan hci_dev_lock(hdev); 9005e762444SAntti Julku 901dcc36c16SJohan Hedberg err = hci_bdaddr_list_del(&hdev->blacklist, &bdaddr, BDADDR_BREDR); 9025e762444SAntti Julku 90309fd0de5SGustavo F. Padovan hci_dev_unlock(hdev); 9045e762444SAntti Julku 9055e762444SAntti Julku return err; 906f0358568SJohan Hedberg } 907f0358568SJohan Hedberg 9081da177e4SLinus Torvalds /* Ioctls that require bound socket */ 9096039aa73SGustavo Padovan static int hci_sock_bound_ioctl(struct sock *sk, unsigned int cmd, 9106039aa73SGustavo Padovan unsigned long arg) 9111da177e4SLinus Torvalds { 9121da177e4SLinus Torvalds struct hci_dev *hdev = hci_pi(sk)->hdev; 9131da177e4SLinus Torvalds 9141da177e4SLinus Torvalds if (!hdev) 9151da177e4SLinus Torvalds return -EBADFD; 9161da177e4SLinus Torvalds 917d7a5a11dSMarcel Holtmann if (hci_dev_test_flag(hdev, HCI_USER_CHANNEL)) 9180736cfa8SMarcel Holtmann return -EBUSY; 9190736cfa8SMarcel Holtmann 920d7a5a11dSMarcel Holtmann if (hci_dev_test_flag(hdev, HCI_UNCONFIGURED)) 921fee746b0SMarcel Holtmann return -EOPNOTSUPP; 922fee746b0SMarcel Holtmann 923ca8bee5dSMarcel Holtmann if (hdev->dev_type != HCI_PRIMARY) 9245b69bef5SMarcel Holtmann return -EOPNOTSUPP; 9255b69bef5SMarcel Holtmann 9261da177e4SLinus Torvalds switch (cmd) { 9271da177e4SLinus Torvalds case HCISETRAW: 9281da177e4SLinus Torvalds if (!capable(CAP_NET_ADMIN)) 929bf5b30b8SZhao Hongjiang return -EPERM; 930db596681SMarcel Holtmann return -EOPNOTSUPP; 9311da177e4SLinus Torvalds 9321da177e4SLinus Torvalds case HCIGETCONNINFO: 9331da177e4SLinus Torvalds return hci_get_conn_info(hdev, (void __user *)arg); 9341da177e4SLinus Torvalds 93540be492fSMarcel Holtmann case HCIGETAUTHINFO: 93640be492fSMarcel Holtmann return hci_get_auth_info(hdev, (void __user *)arg); 93740be492fSMarcel Holtmann 938f0358568SJohan Hedberg case HCIBLOCKADDR: 939f0358568SJohan Hedberg if (!capable(CAP_NET_ADMIN)) 940bf5b30b8SZhao Hongjiang return -EPERM; 941b2a66aadSAntti Julku return hci_sock_blacklist_add(hdev, (void __user *)arg); 942f0358568SJohan Hedberg 943f0358568SJohan Hedberg case HCIUNBLOCKADDR: 944f0358568SJohan Hedberg if (!capable(CAP_NET_ADMIN)) 945bf5b30b8SZhao Hongjiang return -EPERM; 946b2a66aadSAntti Julku return hci_sock_blacklist_del(hdev, (void __user *)arg); 9470736cfa8SMarcel Holtmann } 948f0358568SJohan Hedberg 949324d36edSMarcel Holtmann return -ENOIOCTLCMD; 9501da177e4SLinus Torvalds } 9511da177e4SLinus Torvalds 9528fc9ced3SGustavo Padovan static int hci_sock_ioctl(struct socket *sock, unsigned int cmd, 9538fc9ced3SGustavo Padovan unsigned long arg) 9541da177e4SLinus Torvalds { 9551da177e4SLinus Torvalds void __user *argp = (void __user *)arg; 9560736cfa8SMarcel Holtmann struct sock *sk = sock->sk; 9571da177e4SLinus Torvalds int err; 9581da177e4SLinus Torvalds 9591da177e4SLinus Torvalds BT_DBG("cmd %x arg %lx", cmd, arg); 9601da177e4SLinus Torvalds 961c1c4f956SMarcel Holtmann lock_sock(sk); 962c1c4f956SMarcel Holtmann 963c1c4f956SMarcel Holtmann if (hci_pi(sk)->channel != HCI_CHANNEL_RAW) { 964c1c4f956SMarcel Holtmann err = -EBADFD; 965c1c4f956SMarcel Holtmann goto done; 966c1c4f956SMarcel Holtmann } 967c1c4f956SMarcel Holtmann 968*f81f5b2dSMarcel Holtmann /* When calling an ioctl on an unbound raw socket, then ensure 969*f81f5b2dSMarcel Holtmann * that the monitor gets informed. Ensure that the resulting event 970*f81f5b2dSMarcel Holtmann * is only send once by checking if the cookie exists or not. The 971*f81f5b2dSMarcel Holtmann * socket cookie will be only ever generated once for the lifetime 972*f81f5b2dSMarcel Holtmann * of a given socket. 973*f81f5b2dSMarcel Holtmann */ 974*f81f5b2dSMarcel Holtmann if (hci_sock_gen_cookie(sk)) { 975*f81f5b2dSMarcel Holtmann struct sk_buff *skb; 976*f81f5b2dSMarcel Holtmann 977*f81f5b2dSMarcel Holtmann if (capable(CAP_NET_ADMIN)) 978*f81f5b2dSMarcel Holtmann hci_sock_set_flag(sk, HCI_SOCK_TRUSTED); 979*f81f5b2dSMarcel Holtmann 980*f81f5b2dSMarcel Holtmann /* Send event to monitor */ 981*f81f5b2dSMarcel Holtmann skb = create_monitor_ctrl_open(sk); 982*f81f5b2dSMarcel Holtmann if (skb) { 983*f81f5b2dSMarcel Holtmann hci_send_to_channel(HCI_CHANNEL_MONITOR, skb, 984*f81f5b2dSMarcel Holtmann HCI_SOCK_TRUSTED, NULL); 985*f81f5b2dSMarcel Holtmann kfree_skb(skb); 986*f81f5b2dSMarcel Holtmann } 987*f81f5b2dSMarcel Holtmann } 988*f81f5b2dSMarcel Holtmann 989c1c4f956SMarcel Holtmann release_sock(sk); 990c1c4f956SMarcel Holtmann 9911da177e4SLinus Torvalds switch (cmd) { 9921da177e4SLinus Torvalds case HCIGETDEVLIST: 9931da177e4SLinus Torvalds return hci_get_dev_list(argp); 9941da177e4SLinus Torvalds 9951da177e4SLinus Torvalds case HCIGETDEVINFO: 9961da177e4SLinus Torvalds return hci_get_dev_info(argp); 9971da177e4SLinus Torvalds 9981da177e4SLinus Torvalds case HCIGETCONNLIST: 9991da177e4SLinus Torvalds return hci_get_conn_list(argp); 10001da177e4SLinus Torvalds 10011da177e4SLinus Torvalds case HCIDEVUP: 10021da177e4SLinus Torvalds if (!capable(CAP_NET_ADMIN)) 1003bf5b30b8SZhao Hongjiang return -EPERM; 10041da177e4SLinus Torvalds return hci_dev_open(arg); 10051da177e4SLinus Torvalds 10061da177e4SLinus Torvalds case HCIDEVDOWN: 10071da177e4SLinus Torvalds if (!capable(CAP_NET_ADMIN)) 1008bf5b30b8SZhao Hongjiang return -EPERM; 10091da177e4SLinus Torvalds return hci_dev_close(arg); 10101da177e4SLinus Torvalds 10111da177e4SLinus Torvalds case HCIDEVRESET: 10121da177e4SLinus Torvalds if (!capable(CAP_NET_ADMIN)) 1013bf5b30b8SZhao Hongjiang return -EPERM; 10141da177e4SLinus Torvalds return hci_dev_reset(arg); 10151da177e4SLinus Torvalds 10161da177e4SLinus Torvalds case HCIDEVRESTAT: 10171da177e4SLinus Torvalds if (!capable(CAP_NET_ADMIN)) 1018bf5b30b8SZhao Hongjiang return -EPERM; 10191da177e4SLinus Torvalds return hci_dev_reset_stat(arg); 10201da177e4SLinus Torvalds 10211da177e4SLinus Torvalds case HCISETSCAN: 10221da177e4SLinus Torvalds case HCISETAUTH: 10231da177e4SLinus Torvalds case HCISETENCRYPT: 10241da177e4SLinus Torvalds case HCISETPTYPE: 10251da177e4SLinus Torvalds case HCISETLINKPOL: 10261da177e4SLinus Torvalds case HCISETLINKMODE: 10271da177e4SLinus Torvalds case HCISETACLMTU: 10281da177e4SLinus Torvalds case HCISETSCOMTU: 10291da177e4SLinus Torvalds if (!capable(CAP_NET_ADMIN)) 1030bf5b30b8SZhao Hongjiang return -EPERM; 10311da177e4SLinus Torvalds return hci_dev_cmd(cmd, argp); 10321da177e4SLinus Torvalds 10331da177e4SLinus Torvalds case HCIINQUIRY: 10341da177e4SLinus Torvalds return hci_inquiry(argp); 1035c1c4f956SMarcel Holtmann } 10361da177e4SLinus Torvalds 10371da177e4SLinus Torvalds lock_sock(sk); 1038c1c4f956SMarcel Holtmann 10391da177e4SLinus Torvalds err = hci_sock_bound_ioctl(sk, cmd, arg); 1040c1c4f956SMarcel Holtmann 1041c1c4f956SMarcel Holtmann done: 10421da177e4SLinus Torvalds release_sock(sk); 10431da177e4SLinus Torvalds return err; 10441da177e4SLinus Torvalds } 10451da177e4SLinus Torvalds 10468fc9ced3SGustavo Padovan static int hci_sock_bind(struct socket *sock, struct sockaddr *addr, 10478fc9ced3SGustavo Padovan int addr_len) 10481da177e4SLinus Torvalds { 10490381101fSJohan Hedberg struct sockaddr_hci haddr; 10501da177e4SLinus Torvalds struct sock *sk = sock->sk; 10511da177e4SLinus Torvalds struct hci_dev *hdev = NULL; 10520381101fSJohan Hedberg int len, err = 0; 10531da177e4SLinus Torvalds 10541da177e4SLinus Torvalds BT_DBG("sock %p sk %p", sock, sk); 10551da177e4SLinus Torvalds 10560381101fSJohan Hedberg if (!addr) 10570381101fSJohan Hedberg return -EINVAL; 10580381101fSJohan Hedberg 10590381101fSJohan Hedberg memset(&haddr, 0, sizeof(haddr)); 10600381101fSJohan Hedberg len = min_t(unsigned int, sizeof(haddr), addr_len); 10610381101fSJohan Hedberg memcpy(&haddr, addr, len); 10620381101fSJohan Hedberg 10630381101fSJohan Hedberg if (haddr.hci_family != AF_BLUETOOTH) 10640381101fSJohan Hedberg return -EINVAL; 10650381101fSJohan Hedberg 10661da177e4SLinus Torvalds lock_sock(sk); 10671da177e4SLinus Torvalds 10687cc2ade2SMarcel Holtmann if (sk->sk_state == BT_BOUND) { 10697cc2ade2SMarcel Holtmann err = -EALREADY; 10707cc2ade2SMarcel Holtmann goto done; 10717cc2ade2SMarcel Holtmann } 10727cc2ade2SMarcel Holtmann 10737cc2ade2SMarcel Holtmann switch (haddr.hci_channel) { 10747cc2ade2SMarcel Holtmann case HCI_CHANNEL_RAW: 10757cc2ade2SMarcel Holtmann if (hci_pi(sk)->hdev) { 10761da177e4SLinus Torvalds err = -EALREADY; 10771da177e4SLinus Torvalds goto done; 10781da177e4SLinus Torvalds } 10791da177e4SLinus Torvalds 10800381101fSJohan Hedberg if (haddr.hci_dev != HCI_DEV_NONE) { 10810381101fSJohan Hedberg hdev = hci_dev_get(haddr.hci_dev); 108270f23020SAndrei Emeltchenko if (!hdev) { 10831da177e4SLinus Torvalds err = -ENODEV; 10841da177e4SLinus Torvalds goto done; 10851da177e4SLinus Torvalds } 10861da177e4SLinus Torvalds 10871da177e4SLinus Torvalds atomic_inc(&hdev->promisc); 10881da177e4SLinus Torvalds } 10891da177e4SLinus Torvalds 10905a6d2cf5SMarcel Holtmann hci_pi(sk)->channel = haddr.hci_channel; 10911da177e4SLinus Torvalds hci_pi(sk)->hdev = hdev; 1092*f81f5b2dSMarcel Holtmann 1093*f81f5b2dSMarcel Holtmann /* Only send the event to monitor when a new cookie has 1094*f81f5b2dSMarcel Holtmann * been generated. An existing cookie means that an unbound 1095*f81f5b2dSMarcel Holtmann * socket has seen an ioctl and that triggered the cookie 1096*f81f5b2dSMarcel Holtmann * generation and sending of the monitor event. 1097*f81f5b2dSMarcel Holtmann */ 1098*f81f5b2dSMarcel Holtmann if (hci_sock_gen_cookie(sk)) { 1099*f81f5b2dSMarcel Holtmann struct sk_buff *skb; 1100*f81f5b2dSMarcel Holtmann 1101*f81f5b2dSMarcel Holtmann if (capable(CAP_NET_ADMIN)) 1102*f81f5b2dSMarcel Holtmann hci_sock_set_flag(sk, HCI_SOCK_TRUSTED); 1103*f81f5b2dSMarcel Holtmann 1104*f81f5b2dSMarcel Holtmann /* Send event to monitor */ 1105*f81f5b2dSMarcel Holtmann skb = create_monitor_ctrl_open(sk); 1106*f81f5b2dSMarcel Holtmann if (skb) { 1107*f81f5b2dSMarcel Holtmann hci_send_to_channel(HCI_CHANNEL_MONITOR, skb, 1108*f81f5b2dSMarcel Holtmann HCI_SOCK_TRUSTED, NULL); 1109*f81f5b2dSMarcel Holtmann kfree_skb(skb); 1110*f81f5b2dSMarcel Holtmann } 1111*f81f5b2dSMarcel Holtmann } 11127cc2ade2SMarcel Holtmann break; 11137cc2ade2SMarcel Holtmann 111423500189SMarcel Holtmann case HCI_CHANNEL_USER: 111523500189SMarcel Holtmann if (hci_pi(sk)->hdev) { 111623500189SMarcel Holtmann err = -EALREADY; 111723500189SMarcel Holtmann goto done; 111823500189SMarcel Holtmann } 111923500189SMarcel Holtmann 112023500189SMarcel Holtmann if (haddr.hci_dev == HCI_DEV_NONE) { 112123500189SMarcel Holtmann err = -EINVAL; 112223500189SMarcel Holtmann goto done; 112323500189SMarcel Holtmann } 112423500189SMarcel Holtmann 112510a8b86fSMarcel Holtmann if (!capable(CAP_NET_ADMIN)) { 112623500189SMarcel Holtmann err = -EPERM; 112723500189SMarcel Holtmann goto done; 112823500189SMarcel Holtmann } 112923500189SMarcel Holtmann 113023500189SMarcel Holtmann hdev = hci_dev_get(haddr.hci_dev); 113123500189SMarcel Holtmann if (!hdev) { 113223500189SMarcel Holtmann err = -ENODEV; 113323500189SMarcel Holtmann goto done; 113423500189SMarcel Holtmann } 113523500189SMarcel Holtmann 1136781f899fSMarcel Holtmann if (test_bit(HCI_INIT, &hdev->flags) || 1137d7a5a11dSMarcel Holtmann hci_dev_test_flag(hdev, HCI_SETUP) || 1138781f899fSMarcel Holtmann hci_dev_test_flag(hdev, HCI_CONFIG) || 1139781f899fSMarcel Holtmann (!hci_dev_test_flag(hdev, HCI_AUTO_OFF) && 1140781f899fSMarcel Holtmann test_bit(HCI_UP, &hdev->flags))) { 114123500189SMarcel Holtmann err = -EBUSY; 114223500189SMarcel Holtmann hci_dev_put(hdev); 114323500189SMarcel Holtmann goto done; 114423500189SMarcel Holtmann } 114523500189SMarcel Holtmann 1146238be788SMarcel Holtmann if (hci_dev_test_and_set_flag(hdev, HCI_USER_CHANNEL)) { 114723500189SMarcel Holtmann err = -EUSERS; 114823500189SMarcel Holtmann hci_dev_put(hdev); 114923500189SMarcel Holtmann goto done; 115023500189SMarcel Holtmann } 115123500189SMarcel Holtmann 115223500189SMarcel Holtmann mgmt_index_removed(hdev); 115323500189SMarcel Holtmann 115423500189SMarcel Holtmann err = hci_dev_open(hdev->id); 115523500189SMarcel Holtmann if (err) { 1156781f899fSMarcel Holtmann if (err == -EALREADY) { 1157781f899fSMarcel Holtmann /* In case the transport is already up and 1158781f899fSMarcel Holtmann * running, clear the error here. 1159781f899fSMarcel Holtmann * 1160781f899fSMarcel Holtmann * This can happen when opening an user 1161781f899fSMarcel Holtmann * channel and HCI_AUTO_OFF grace period 1162781f899fSMarcel Holtmann * is still active. 1163781f899fSMarcel Holtmann */ 1164781f899fSMarcel Holtmann err = 0; 1165781f899fSMarcel Holtmann } else { 1166a358dc11SMarcel Holtmann hci_dev_clear_flag(hdev, HCI_USER_CHANNEL); 1167c6521401SMarcel Holtmann mgmt_index_added(hdev); 116823500189SMarcel Holtmann hci_dev_put(hdev); 116923500189SMarcel Holtmann goto done; 117023500189SMarcel Holtmann } 1171781f899fSMarcel Holtmann } 117223500189SMarcel Holtmann 11735a6d2cf5SMarcel Holtmann hci_pi(sk)->channel = haddr.hci_channel; 117423500189SMarcel Holtmann hci_pi(sk)->hdev = hdev; 11755a6d2cf5SMarcel Holtmann 11765a6d2cf5SMarcel Holtmann atomic_inc(&hdev->promisc); 117723500189SMarcel Holtmann break; 117823500189SMarcel Holtmann 1179cd82e61cSMarcel Holtmann case HCI_CHANNEL_MONITOR: 1180cd82e61cSMarcel Holtmann if (haddr.hci_dev != HCI_DEV_NONE) { 1181cd82e61cSMarcel Holtmann err = -EINVAL; 1182cd82e61cSMarcel Holtmann goto done; 1183cd82e61cSMarcel Holtmann } 1184cd82e61cSMarcel Holtmann 1185cd82e61cSMarcel Holtmann if (!capable(CAP_NET_RAW)) { 1186cd82e61cSMarcel Holtmann err = -EPERM; 1187cd82e61cSMarcel Holtmann goto done; 1188cd82e61cSMarcel Holtmann } 1189cd82e61cSMarcel Holtmann 11905a6d2cf5SMarcel Holtmann hci_pi(sk)->channel = haddr.hci_channel; 11915a6d2cf5SMarcel Holtmann 119250ebc055SMarcel Holtmann /* The monitor interface is restricted to CAP_NET_RAW 119350ebc055SMarcel Holtmann * capabilities and with that implicitly trusted. 119450ebc055SMarcel Holtmann */ 119550ebc055SMarcel Holtmann hci_sock_set_flag(sk, HCI_SOCK_TRUSTED); 119650ebc055SMarcel Holtmann 1197787b306cSJohannes Berg send_monitor_note(sk, "Linux version %s (%s)", 1198787b306cSJohannes Berg init_utsname()->release, 1199787b306cSJohannes Berg init_utsname()->machine); 12009e8305b3SMarcel Holtmann send_monitor_note(sk, "Bluetooth subsystem version %u.%u", 12019e8305b3SMarcel Holtmann BT_SUBSYS_VERSION, BT_SUBSYS_REVISION); 1202cd82e61cSMarcel Holtmann send_monitor_replay(sk); 1203249fa169SMarcel Holtmann send_monitor_control_replay(sk); 1204cd82e61cSMarcel Holtmann 1205cd82e61cSMarcel Holtmann atomic_inc(&monitor_promisc); 1206cd82e61cSMarcel Holtmann break; 1207cd82e61cSMarcel Holtmann 1208ac714949SMarcel Holtmann case HCI_CHANNEL_LOGGING: 1209ac714949SMarcel Holtmann if (haddr.hci_dev != HCI_DEV_NONE) { 1210ac714949SMarcel Holtmann err = -EINVAL; 1211ac714949SMarcel Holtmann goto done; 1212ac714949SMarcel Holtmann } 1213ac714949SMarcel Holtmann 1214ac714949SMarcel Holtmann if (!capable(CAP_NET_ADMIN)) { 1215ac714949SMarcel Holtmann err = -EPERM; 1216ac714949SMarcel Holtmann goto done; 1217ac714949SMarcel Holtmann } 12185a6d2cf5SMarcel Holtmann 12195a6d2cf5SMarcel Holtmann hci_pi(sk)->channel = haddr.hci_channel; 1220ac714949SMarcel Holtmann break; 1221ac714949SMarcel Holtmann 12227cc2ade2SMarcel Holtmann default: 1223801c1e8dSJohan Hedberg if (!hci_mgmt_chan_find(haddr.hci_channel)) { 12247cc2ade2SMarcel Holtmann err = -EINVAL; 12257cc2ade2SMarcel Holtmann goto done; 12267cc2ade2SMarcel Holtmann } 12277cc2ade2SMarcel Holtmann 1228801c1e8dSJohan Hedberg if (haddr.hci_dev != HCI_DEV_NONE) { 1229801c1e8dSJohan Hedberg err = -EINVAL; 1230801c1e8dSJohan Hedberg goto done; 1231801c1e8dSJohan Hedberg } 1232801c1e8dSJohan Hedberg 12331195fbb8SMarcel Holtmann /* Users with CAP_NET_ADMIN capabilities are allowed 12341195fbb8SMarcel Holtmann * access to all management commands and events. For 12351195fbb8SMarcel Holtmann * untrusted users the interface is restricted and 12361195fbb8SMarcel Holtmann * also only untrusted events are sent. 123750ebc055SMarcel Holtmann */ 12381195fbb8SMarcel Holtmann if (capable(CAP_NET_ADMIN)) 123950ebc055SMarcel Holtmann hci_sock_set_flag(sk, HCI_SOCK_TRUSTED); 124050ebc055SMarcel Holtmann 12415a6d2cf5SMarcel Holtmann hci_pi(sk)->channel = haddr.hci_channel; 12425a6d2cf5SMarcel Holtmann 1243f9207338SMarcel Holtmann /* At the moment the index and unconfigured index events 1244f9207338SMarcel Holtmann * are enabled unconditionally. Setting them on each 1245f9207338SMarcel Holtmann * socket when binding keeps this functionality. They 1246f9207338SMarcel Holtmann * however might be cleared later and then sending of these 1247f9207338SMarcel Holtmann * events will be disabled, but that is then intentional. 1248f6b7712eSMarcel Holtmann * 1249f6b7712eSMarcel Holtmann * This also enables generic events that are safe to be 1250f6b7712eSMarcel Holtmann * received by untrusted users. Example for such events 1251f6b7712eSMarcel Holtmann * are changes to settings, class of device, name etc. 1252f9207338SMarcel Holtmann */ 12535a6d2cf5SMarcel Holtmann if (hci_pi(sk)->channel == HCI_CHANNEL_CONTROL) { 1254249fa169SMarcel Holtmann struct sk_buff *skb; 125570ecce91SMarcel Holtmann 1256df1cb87aSMarcel Holtmann hci_sock_gen_cookie(sk); 125770ecce91SMarcel Holtmann 1258249fa169SMarcel Holtmann /* Send event to monitor */ 1259249fa169SMarcel Holtmann skb = create_monitor_ctrl_open(sk); 1260249fa169SMarcel Holtmann if (skb) { 1261249fa169SMarcel Holtmann hci_send_to_channel(HCI_CHANNEL_MONITOR, skb, 1262249fa169SMarcel Holtmann HCI_SOCK_TRUSTED, NULL); 1263249fa169SMarcel Holtmann kfree_skb(skb); 1264249fa169SMarcel Holtmann } 1265249fa169SMarcel Holtmann 1266f9207338SMarcel Holtmann hci_sock_set_flag(sk, HCI_MGMT_INDEX_EVENTS); 1267f9207338SMarcel Holtmann hci_sock_set_flag(sk, HCI_MGMT_UNCONF_INDEX_EVENTS); 12685504c3a3SMarcel Holtmann hci_sock_set_flag(sk, HCI_MGMT_OPTION_EVENTS); 12695504c3a3SMarcel Holtmann hci_sock_set_flag(sk, HCI_MGMT_SETTING_EVENTS); 12705504c3a3SMarcel Holtmann hci_sock_set_flag(sk, HCI_MGMT_DEV_CLASS_EVENTS); 12715504c3a3SMarcel Holtmann hci_sock_set_flag(sk, HCI_MGMT_LOCAL_NAME_EVENTS); 1272f9207338SMarcel Holtmann } 1273801c1e8dSJohan Hedberg break; 1274801c1e8dSJohan Hedberg } 1275801c1e8dSJohan Hedberg 12761da177e4SLinus Torvalds sk->sk_state = BT_BOUND; 12771da177e4SLinus Torvalds 12781da177e4SLinus Torvalds done: 12791da177e4SLinus Torvalds release_sock(sk); 12801da177e4SLinus Torvalds return err; 12811da177e4SLinus Torvalds } 12821da177e4SLinus Torvalds 12838fc9ced3SGustavo Padovan static int hci_sock_getname(struct socket *sock, struct sockaddr *addr, 12848fc9ced3SGustavo Padovan int *addr_len, int peer) 12851da177e4SLinus Torvalds { 12861da177e4SLinus Torvalds struct sockaddr_hci *haddr = (struct sockaddr_hci *)addr; 12871da177e4SLinus Torvalds struct sock *sk = sock->sk; 12889d4b68b2SMarcel Holtmann struct hci_dev *hdev; 12899d4b68b2SMarcel Holtmann int err = 0; 12901da177e4SLinus Torvalds 12911da177e4SLinus Torvalds BT_DBG("sock %p sk %p", sock, sk); 12921da177e4SLinus Torvalds 129306f43cbcSMarcel Holtmann if (peer) 129406f43cbcSMarcel Holtmann return -EOPNOTSUPP; 129506f43cbcSMarcel Holtmann 12961da177e4SLinus Torvalds lock_sock(sk); 12971da177e4SLinus Torvalds 12989d4b68b2SMarcel Holtmann hdev = hci_pi(sk)->hdev; 12999d4b68b2SMarcel Holtmann if (!hdev) { 13009d4b68b2SMarcel Holtmann err = -EBADFD; 13019d4b68b2SMarcel Holtmann goto done; 13029d4b68b2SMarcel Holtmann } 13039d4b68b2SMarcel Holtmann 13041da177e4SLinus Torvalds *addr_len = sizeof(*haddr); 13051da177e4SLinus Torvalds haddr->hci_family = AF_BLUETOOTH; 13067b005bd3SMarcel Holtmann haddr->hci_dev = hdev->id; 13079d4b68b2SMarcel Holtmann haddr->hci_channel= hci_pi(sk)->channel; 13081da177e4SLinus Torvalds 13099d4b68b2SMarcel Holtmann done: 13101da177e4SLinus Torvalds release_sock(sk); 13119d4b68b2SMarcel Holtmann return err; 13121da177e4SLinus Torvalds } 13131da177e4SLinus Torvalds 13146039aa73SGustavo Padovan static void hci_sock_cmsg(struct sock *sk, struct msghdr *msg, 13156039aa73SGustavo Padovan struct sk_buff *skb) 13161da177e4SLinus Torvalds { 13171da177e4SLinus Torvalds __u32 mask = hci_pi(sk)->cmsg_mask; 13181da177e4SLinus Torvalds 13190d48d939SMarcel Holtmann if (mask & HCI_CMSG_DIR) { 13200d48d939SMarcel Holtmann int incoming = bt_cb(skb)->incoming; 13218fc9ced3SGustavo Padovan put_cmsg(msg, SOL_HCI, HCI_CMSG_DIR, sizeof(incoming), 13228fc9ced3SGustavo Padovan &incoming); 13230d48d939SMarcel Holtmann } 13241da177e4SLinus Torvalds 1325a61bbcf2SPatrick McHardy if (mask & HCI_CMSG_TSTAMP) { 1326f6e623a6SJohann Felix Soden #ifdef CONFIG_COMPAT 1327f6e623a6SJohann Felix Soden struct compat_timeval ctv; 1328f6e623a6SJohann Felix Soden #endif 1329a61bbcf2SPatrick McHardy struct timeval tv; 1330767c5eb5SMarcel Holtmann void *data; 1331767c5eb5SMarcel Holtmann int len; 1332a61bbcf2SPatrick McHardy 1333a61bbcf2SPatrick McHardy skb_get_timestamp(skb, &tv); 1334767c5eb5SMarcel Holtmann 13351da97f83SDavid S. Miller data = &tv; 13361da97f83SDavid S. Miller len = sizeof(tv); 13371da97f83SDavid S. Miller #ifdef CONFIG_COMPAT 1338da88cea1SH. J. Lu if (!COMPAT_USE_64BIT_TIME && 1339da88cea1SH. J. Lu (msg->msg_flags & MSG_CMSG_COMPAT)) { 1340767c5eb5SMarcel Holtmann ctv.tv_sec = tv.tv_sec; 1341767c5eb5SMarcel Holtmann ctv.tv_usec = tv.tv_usec; 1342767c5eb5SMarcel Holtmann data = &ctv; 1343767c5eb5SMarcel Holtmann len = sizeof(ctv); 1344767c5eb5SMarcel Holtmann } 13451da97f83SDavid S. Miller #endif 1346767c5eb5SMarcel Holtmann 1347767c5eb5SMarcel Holtmann put_cmsg(msg, SOL_HCI, HCI_CMSG_TSTAMP, len, data); 1348a61bbcf2SPatrick McHardy } 13491da177e4SLinus Torvalds } 13501da177e4SLinus Torvalds 13518528d3f7SMarcel Holtmann static int hci_sock_recvmsg(struct socket *sock, struct msghdr *msg, 13528528d3f7SMarcel Holtmann size_t len, int flags) 13531da177e4SLinus Torvalds { 13541da177e4SLinus Torvalds int noblock = flags & MSG_DONTWAIT; 13551da177e4SLinus Torvalds struct sock *sk = sock->sk; 13561da177e4SLinus Torvalds struct sk_buff *skb; 13571da177e4SLinus Torvalds int copied, err; 135883871f8cSDenis Kenzior unsigned int skblen; 13591da177e4SLinus Torvalds 13601da177e4SLinus Torvalds BT_DBG("sock %p, sk %p", sock, sk); 13611da177e4SLinus Torvalds 1362d94a6104SMarcel Holtmann if (flags & MSG_OOB) 13631da177e4SLinus Torvalds return -EOPNOTSUPP; 13641da177e4SLinus Torvalds 1365ac714949SMarcel Holtmann if (hci_pi(sk)->channel == HCI_CHANNEL_LOGGING) 1366ac714949SMarcel Holtmann return -EOPNOTSUPP; 1367ac714949SMarcel Holtmann 13681da177e4SLinus Torvalds if (sk->sk_state == BT_CLOSED) 13691da177e4SLinus Torvalds return 0; 13701da177e4SLinus Torvalds 137170f23020SAndrei Emeltchenko skb = skb_recv_datagram(sk, flags, noblock, &err); 137270f23020SAndrei Emeltchenko if (!skb) 13731da177e4SLinus Torvalds return err; 13741da177e4SLinus Torvalds 137583871f8cSDenis Kenzior skblen = skb->len; 13761da177e4SLinus Torvalds copied = skb->len; 13771da177e4SLinus Torvalds if (len < copied) { 13781da177e4SLinus Torvalds msg->msg_flags |= MSG_TRUNC; 13791da177e4SLinus Torvalds copied = len; 13801da177e4SLinus Torvalds } 13811da177e4SLinus Torvalds 1382badff6d0SArnaldo Carvalho de Melo skb_reset_transport_header(skb); 138351f3d02bSDavid S. Miller err = skb_copy_datagram_msg(skb, 0, msg, copied); 13841da177e4SLinus Torvalds 13853a208627SMarcel Holtmann switch (hci_pi(sk)->channel) { 13863a208627SMarcel Holtmann case HCI_CHANNEL_RAW: 13871da177e4SLinus Torvalds hci_sock_cmsg(sk, msg, skb); 13883a208627SMarcel Holtmann break; 138923500189SMarcel Holtmann case HCI_CHANNEL_USER: 1390cd82e61cSMarcel Holtmann case HCI_CHANNEL_MONITOR: 1391cd82e61cSMarcel Holtmann sock_recv_timestamp(msg, sk, skb); 1392cd82e61cSMarcel Holtmann break; 1393801c1e8dSJohan Hedberg default: 1394801c1e8dSJohan Hedberg if (hci_mgmt_chan_find(hci_pi(sk)->channel)) 1395801c1e8dSJohan Hedberg sock_recv_timestamp(msg, sk, skb); 1396801c1e8dSJohan Hedberg break; 13973a208627SMarcel Holtmann } 13981da177e4SLinus Torvalds 13991da177e4SLinus Torvalds skb_free_datagram(sk, skb); 14001da177e4SLinus Torvalds 14014f34228bSLuiz Augusto von Dentz if (flags & MSG_TRUNC) 140283871f8cSDenis Kenzior copied = skblen; 140383871f8cSDenis Kenzior 14041da177e4SLinus Torvalds return err ? : copied; 14051da177e4SLinus Torvalds } 14061da177e4SLinus Torvalds 1407fa4335d7SJohan Hedberg static int hci_mgmt_cmd(struct hci_mgmt_chan *chan, struct sock *sk, 1408fa4335d7SJohan Hedberg struct msghdr *msg, size_t msglen) 1409fa4335d7SJohan Hedberg { 1410fa4335d7SJohan Hedberg void *buf; 1411fa4335d7SJohan Hedberg u8 *cp; 1412fa4335d7SJohan Hedberg struct mgmt_hdr *hdr; 1413fa4335d7SJohan Hedberg u16 opcode, index, len; 1414fa4335d7SJohan Hedberg struct hci_dev *hdev = NULL; 1415fa4335d7SJohan Hedberg const struct hci_mgmt_handler *handler; 1416fa4335d7SJohan Hedberg bool var_len, no_hdev; 1417fa4335d7SJohan Hedberg int err; 1418fa4335d7SJohan Hedberg 1419fa4335d7SJohan Hedberg BT_DBG("got %zu bytes", msglen); 1420fa4335d7SJohan Hedberg 1421fa4335d7SJohan Hedberg if (msglen < sizeof(*hdr)) 1422fa4335d7SJohan Hedberg return -EINVAL; 1423fa4335d7SJohan Hedberg 1424fa4335d7SJohan Hedberg buf = kmalloc(msglen, GFP_KERNEL); 1425fa4335d7SJohan Hedberg if (!buf) 1426fa4335d7SJohan Hedberg return -ENOMEM; 1427fa4335d7SJohan Hedberg 1428fa4335d7SJohan Hedberg if (memcpy_from_msg(buf, msg, msglen)) { 1429fa4335d7SJohan Hedberg err = -EFAULT; 1430fa4335d7SJohan Hedberg goto done; 1431fa4335d7SJohan Hedberg } 1432fa4335d7SJohan Hedberg 1433fa4335d7SJohan Hedberg hdr = buf; 1434fa4335d7SJohan Hedberg opcode = __le16_to_cpu(hdr->opcode); 1435fa4335d7SJohan Hedberg index = __le16_to_cpu(hdr->index); 1436fa4335d7SJohan Hedberg len = __le16_to_cpu(hdr->len); 1437fa4335d7SJohan Hedberg 1438fa4335d7SJohan Hedberg if (len != msglen - sizeof(*hdr)) { 1439fa4335d7SJohan Hedberg err = -EINVAL; 1440fa4335d7SJohan Hedberg goto done; 1441fa4335d7SJohan Hedberg } 1442fa4335d7SJohan Hedberg 144338ceaa00SMarcel Holtmann if (chan->channel == HCI_CHANNEL_CONTROL) { 144438ceaa00SMarcel Holtmann struct sk_buff *skb; 144538ceaa00SMarcel Holtmann 144638ceaa00SMarcel Holtmann /* Send event to monitor */ 144738ceaa00SMarcel Holtmann skb = create_monitor_ctrl_command(sk, index, opcode, len, 144838ceaa00SMarcel Holtmann buf + sizeof(*hdr)); 144938ceaa00SMarcel Holtmann if (skb) { 145038ceaa00SMarcel Holtmann hci_send_to_channel(HCI_CHANNEL_MONITOR, skb, 145138ceaa00SMarcel Holtmann HCI_SOCK_TRUSTED, NULL); 145238ceaa00SMarcel Holtmann kfree_skb(skb); 145338ceaa00SMarcel Holtmann } 145438ceaa00SMarcel Holtmann } 145538ceaa00SMarcel Holtmann 1456fa4335d7SJohan Hedberg if (opcode >= chan->handler_count || 1457fa4335d7SJohan Hedberg chan->handlers[opcode].func == NULL) { 1458fa4335d7SJohan Hedberg BT_DBG("Unknown op %u", opcode); 1459fa4335d7SJohan Hedberg err = mgmt_cmd_status(sk, index, opcode, 1460fa4335d7SJohan Hedberg MGMT_STATUS_UNKNOWN_COMMAND); 1461fa4335d7SJohan Hedberg goto done; 1462fa4335d7SJohan Hedberg } 1463fa4335d7SJohan Hedberg 1464fa4335d7SJohan Hedberg handler = &chan->handlers[opcode]; 1465fa4335d7SJohan Hedberg 1466fa4335d7SJohan Hedberg if (!hci_sock_test_flag(sk, HCI_SOCK_TRUSTED) && 1467fa4335d7SJohan Hedberg !(handler->flags & HCI_MGMT_UNTRUSTED)) { 1468fa4335d7SJohan Hedberg err = mgmt_cmd_status(sk, index, opcode, 1469fa4335d7SJohan Hedberg MGMT_STATUS_PERMISSION_DENIED); 1470fa4335d7SJohan Hedberg goto done; 1471fa4335d7SJohan Hedberg } 1472fa4335d7SJohan Hedberg 1473fa4335d7SJohan Hedberg if (index != MGMT_INDEX_NONE) { 1474fa4335d7SJohan Hedberg hdev = hci_dev_get(index); 1475fa4335d7SJohan Hedberg if (!hdev) { 1476fa4335d7SJohan Hedberg err = mgmt_cmd_status(sk, index, opcode, 1477fa4335d7SJohan Hedberg MGMT_STATUS_INVALID_INDEX); 1478fa4335d7SJohan Hedberg goto done; 1479fa4335d7SJohan Hedberg } 1480fa4335d7SJohan Hedberg 1481fa4335d7SJohan Hedberg if (hci_dev_test_flag(hdev, HCI_SETUP) || 1482fa4335d7SJohan Hedberg hci_dev_test_flag(hdev, HCI_CONFIG) || 1483fa4335d7SJohan Hedberg hci_dev_test_flag(hdev, HCI_USER_CHANNEL)) { 1484fa4335d7SJohan Hedberg err = mgmt_cmd_status(sk, index, opcode, 1485fa4335d7SJohan Hedberg MGMT_STATUS_INVALID_INDEX); 1486fa4335d7SJohan Hedberg goto done; 1487fa4335d7SJohan Hedberg } 1488fa4335d7SJohan Hedberg 1489fa4335d7SJohan Hedberg if (hci_dev_test_flag(hdev, HCI_UNCONFIGURED) && 1490fa4335d7SJohan Hedberg !(handler->flags & HCI_MGMT_UNCONFIGURED)) { 1491fa4335d7SJohan Hedberg err = mgmt_cmd_status(sk, index, opcode, 1492fa4335d7SJohan Hedberg MGMT_STATUS_INVALID_INDEX); 1493fa4335d7SJohan Hedberg goto done; 1494fa4335d7SJohan Hedberg } 1495fa4335d7SJohan Hedberg } 1496fa4335d7SJohan Hedberg 1497fa4335d7SJohan Hedberg no_hdev = (handler->flags & HCI_MGMT_NO_HDEV); 1498fa4335d7SJohan Hedberg if (no_hdev != !hdev) { 1499fa4335d7SJohan Hedberg err = mgmt_cmd_status(sk, index, opcode, 1500fa4335d7SJohan Hedberg MGMT_STATUS_INVALID_INDEX); 1501fa4335d7SJohan Hedberg goto done; 1502fa4335d7SJohan Hedberg } 1503fa4335d7SJohan Hedberg 1504fa4335d7SJohan Hedberg var_len = (handler->flags & HCI_MGMT_VAR_LEN); 1505fa4335d7SJohan Hedberg if ((var_len && len < handler->data_len) || 1506fa4335d7SJohan Hedberg (!var_len && len != handler->data_len)) { 1507fa4335d7SJohan Hedberg err = mgmt_cmd_status(sk, index, opcode, 1508fa4335d7SJohan Hedberg MGMT_STATUS_INVALID_PARAMS); 1509fa4335d7SJohan Hedberg goto done; 1510fa4335d7SJohan Hedberg } 1511fa4335d7SJohan Hedberg 1512fa4335d7SJohan Hedberg if (hdev && chan->hdev_init) 1513fa4335d7SJohan Hedberg chan->hdev_init(sk, hdev); 1514fa4335d7SJohan Hedberg 1515fa4335d7SJohan Hedberg cp = buf + sizeof(*hdr); 1516fa4335d7SJohan Hedberg 1517fa4335d7SJohan Hedberg err = handler->func(sk, hdev, cp, len); 1518fa4335d7SJohan Hedberg if (err < 0) 1519fa4335d7SJohan Hedberg goto done; 1520fa4335d7SJohan Hedberg 1521fa4335d7SJohan Hedberg err = msglen; 1522fa4335d7SJohan Hedberg 1523fa4335d7SJohan Hedberg done: 1524fa4335d7SJohan Hedberg if (hdev) 1525fa4335d7SJohan Hedberg hci_dev_put(hdev); 1526fa4335d7SJohan Hedberg 1527fa4335d7SJohan Hedberg kfree(buf); 1528fa4335d7SJohan Hedberg return err; 1529fa4335d7SJohan Hedberg } 1530fa4335d7SJohan Hedberg 1531ac714949SMarcel Holtmann static int hci_logging_frame(struct sock *sk, struct msghdr *msg, int len) 1532ac714949SMarcel Holtmann { 1533ac714949SMarcel Holtmann struct hci_mon_hdr *hdr; 1534ac714949SMarcel Holtmann struct sk_buff *skb; 1535ac714949SMarcel Holtmann struct hci_dev *hdev; 1536ac714949SMarcel Holtmann u16 index; 1537ac714949SMarcel Holtmann int err; 1538ac714949SMarcel Holtmann 1539ac714949SMarcel Holtmann /* The logging frame consists at minimum of the standard header, 1540ac714949SMarcel Holtmann * the priority byte, the ident length byte and at least one string 1541ac714949SMarcel Holtmann * terminator NUL byte. Anything shorter are invalid packets. 1542ac714949SMarcel Holtmann */ 1543ac714949SMarcel Holtmann if (len < sizeof(*hdr) + 3) 1544ac714949SMarcel Holtmann return -EINVAL; 1545ac714949SMarcel Holtmann 1546ac714949SMarcel Holtmann skb = bt_skb_send_alloc(sk, len, msg->msg_flags & MSG_DONTWAIT, &err); 1547ac714949SMarcel Holtmann if (!skb) 1548ac714949SMarcel Holtmann return err; 1549ac714949SMarcel Holtmann 1550ac714949SMarcel Holtmann if (memcpy_from_msg(skb_put(skb, len), msg, len)) { 1551ac714949SMarcel Holtmann err = -EFAULT; 1552ac714949SMarcel Holtmann goto drop; 1553ac714949SMarcel Holtmann } 1554ac714949SMarcel Holtmann 1555ac714949SMarcel Holtmann hdr = (void *)skb->data; 1556ac714949SMarcel Holtmann 1557ac714949SMarcel Holtmann if (__le16_to_cpu(hdr->len) != len - sizeof(*hdr)) { 1558ac714949SMarcel Holtmann err = -EINVAL; 1559ac714949SMarcel Holtmann goto drop; 1560ac714949SMarcel Holtmann } 1561ac714949SMarcel Holtmann 1562ac714949SMarcel Holtmann if (__le16_to_cpu(hdr->opcode) == 0x0000) { 1563ac714949SMarcel Holtmann __u8 priority = skb->data[sizeof(*hdr)]; 1564ac714949SMarcel Holtmann __u8 ident_len = skb->data[sizeof(*hdr) + 1]; 1565ac714949SMarcel Holtmann 1566ac714949SMarcel Holtmann /* Only the priorities 0-7 are valid and with that any other 1567ac714949SMarcel Holtmann * value results in an invalid packet. 1568ac714949SMarcel Holtmann * 1569ac714949SMarcel Holtmann * The priority byte is followed by an ident length byte and 1570ac714949SMarcel Holtmann * the NUL terminated ident string. Check that the ident 1571ac714949SMarcel Holtmann * length is not overflowing the packet and also that the 1572ac714949SMarcel Holtmann * ident string itself is NUL terminated. In case the ident 1573ac714949SMarcel Holtmann * length is zero, the length value actually doubles as NUL 1574ac714949SMarcel Holtmann * terminator identifier. 1575ac714949SMarcel Holtmann * 1576ac714949SMarcel Holtmann * The message follows the ident string (if present) and 1577ac714949SMarcel Holtmann * must be NUL terminated. Otherwise it is not a valid packet. 1578ac714949SMarcel Holtmann */ 1579ac714949SMarcel Holtmann if (priority > 7 || skb->data[len - 1] != 0x00 || 1580ac714949SMarcel Holtmann ident_len > len - sizeof(*hdr) - 3 || 1581ac714949SMarcel Holtmann skb->data[sizeof(*hdr) + ident_len + 1] != 0x00) { 1582ac714949SMarcel Holtmann err = -EINVAL; 1583ac714949SMarcel Holtmann goto drop; 1584ac714949SMarcel Holtmann } 1585ac714949SMarcel Holtmann } else { 1586ac714949SMarcel Holtmann err = -EINVAL; 1587ac714949SMarcel Holtmann goto drop; 1588ac714949SMarcel Holtmann } 1589ac714949SMarcel Holtmann 1590ac714949SMarcel Holtmann index = __le16_to_cpu(hdr->index); 1591ac714949SMarcel Holtmann 1592ac714949SMarcel Holtmann if (index != MGMT_INDEX_NONE) { 1593ac714949SMarcel Holtmann hdev = hci_dev_get(index); 1594ac714949SMarcel Holtmann if (!hdev) { 1595ac714949SMarcel Holtmann err = -ENODEV; 1596ac714949SMarcel Holtmann goto drop; 1597ac714949SMarcel Holtmann } 1598ac714949SMarcel Holtmann } else { 1599ac714949SMarcel Holtmann hdev = NULL; 1600ac714949SMarcel Holtmann } 1601ac714949SMarcel Holtmann 1602ac714949SMarcel Holtmann hdr->opcode = cpu_to_le16(HCI_MON_USER_LOGGING); 1603ac714949SMarcel Holtmann 1604ac714949SMarcel Holtmann hci_send_to_channel(HCI_CHANNEL_MONITOR, skb, HCI_SOCK_TRUSTED, NULL); 1605ac714949SMarcel Holtmann err = len; 1606ac714949SMarcel Holtmann 1607ac714949SMarcel Holtmann if (hdev) 1608ac714949SMarcel Holtmann hci_dev_put(hdev); 1609ac714949SMarcel Holtmann 1610ac714949SMarcel Holtmann drop: 1611ac714949SMarcel Holtmann kfree_skb(skb); 1612ac714949SMarcel Holtmann return err; 1613ac714949SMarcel Holtmann } 1614ac714949SMarcel Holtmann 16151b784140SYing Xue static int hci_sock_sendmsg(struct socket *sock, struct msghdr *msg, 16161b784140SYing Xue size_t len) 16171da177e4SLinus Torvalds { 16181da177e4SLinus Torvalds struct sock *sk = sock->sk; 1619801c1e8dSJohan Hedberg struct hci_mgmt_chan *chan; 16201da177e4SLinus Torvalds struct hci_dev *hdev; 16211da177e4SLinus Torvalds struct sk_buff *skb; 16221da177e4SLinus Torvalds int err; 16231da177e4SLinus Torvalds 16241da177e4SLinus Torvalds BT_DBG("sock %p sk %p", sock, sk); 16251da177e4SLinus Torvalds 16261da177e4SLinus Torvalds if (msg->msg_flags & MSG_OOB) 16271da177e4SLinus Torvalds return -EOPNOTSUPP; 16281da177e4SLinus Torvalds 16291da177e4SLinus Torvalds if (msg->msg_flags & ~(MSG_DONTWAIT|MSG_NOSIGNAL|MSG_ERRQUEUE)) 16301da177e4SLinus Torvalds return -EINVAL; 16311da177e4SLinus Torvalds 16321da177e4SLinus Torvalds if (len < 4 || len > HCI_MAX_FRAME_SIZE) 16331da177e4SLinus Torvalds return -EINVAL; 16341da177e4SLinus Torvalds 16351da177e4SLinus Torvalds lock_sock(sk); 16361da177e4SLinus Torvalds 16370381101fSJohan Hedberg switch (hci_pi(sk)->channel) { 16380381101fSJohan Hedberg case HCI_CHANNEL_RAW: 163923500189SMarcel Holtmann case HCI_CHANNEL_USER: 16400381101fSJohan Hedberg break; 1641cd82e61cSMarcel Holtmann case HCI_CHANNEL_MONITOR: 1642cd82e61cSMarcel Holtmann err = -EOPNOTSUPP; 1643cd82e61cSMarcel Holtmann goto done; 1644ac714949SMarcel Holtmann case HCI_CHANNEL_LOGGING: 1645ac714949SMarcel Holtmann err = hci_logging_frame(sk, msg, len); 1646ac714949SMarcel Holtmann goto done; 16470381101fSJohan Hedberg default: 1648801c1e8dSJohan Hedberg mutex_lock(&mgmt_chan_list_lock); 1649801c1e8dSJohan Hedberg chan = __hci_mgmt_chan_find(hci_pi(sk)->channel); 1650801c1e8dSJohan Hedberg if (chan) 1651fa4335d7SJohan Hedberg err = hci_mgmt_cmd(chan, sk, msg, len); 1652801c1e8dSJohan Hedberg else 16530381101fSJohan Hedberg err = -EINVAL; 1654801c1e8dSJohan Hedberg 1655801c1e8dSJohan Hedberg mutex_unlock(&mgmt_chan_list_lock); 16560381101fSJohan Hedberg goto done; 16570381101fSJohan Hedberg } 16580381101fSJohan Hedberg 165970f23020SAndrei Emeltchenko hdev = hci_pi(sk)->hdev; 166070f23020SAndrei Emeltchenko if (!hdev) { 16611da177e4SLinus Torvalds err = -EBADFD; 16621da177e4SLinus Torvalds goto done; 16631da177e4SLinus Torvalds } 16641da177e4SLinus Torvalds 16657e21addcSMarcel Holtmann if (!test_bit(HCI_UP, &hdev->flags)) { 16667e21addcSMarcel Holtmann err = -ENETDOWN; 16677e21addcSMarcel Holtmann goto done; 16687e21addcSMarcel Holtmann } 16697e21addcSMarcel Holtmann 167070f23020SAndrei Emeltchenko skb = bt_skb_send_alloc(sk, len, msg->msg_flags & MSG_DONTWAIT, &err); 167170f23020SAndrei Emeltchenko if (!skb) 16721da177e4SLinus Torvalds goto done; 16731da177e4SLinus Torvalds 16746ce8e9ceSAl Viro if (memcpy_from_msg(skb_put(skb, len), msg, len)) { 16751da177e4SLinus Torvalds err = -EFAULT; 16761da177e4SLinus Torvalds goto drop; 16771da177e4SLinus Torvalds } 16781da177e4SLinus Torvalds 16798528d3f7SMarcel Holtmann hci_skb_pkt_type(skb) = skb->data[0]; 16801da177e4SLinus Torvalds skb_pull(skb, 1); 16811da177e4SLinus Torvalds 16821bc5ad16SMarcel Holtmann if (hci_pi(sk)->channel == HCI_CHANNEL_USER) { 16831bc5ad16SMarcel Holtmann /* No permission check is needed for user channel 16841bc5ad16SMarcel Holtmann * since that gets enforced when binding the socket. 16851bc5ad16SMarcel Holtmann * 16861bc5ad16SMarcel Holtmann * However check that the packet type is valid. 16871bc5ad16SMarcel Holtmann */ 1688d79f34e3SMarcel Holtmann if (hci_skb_pkt_type(skb) != HCI_COMMAND_PKT && 1689d79f34e3SMarcel Holtmann hci_skb_pkt_type(skb) != HCI_ACLDATA_PKT && 1690d79f34e3SMarcel Holtmann hci_skb_pkt_type(skb) != HCI_SCODATA_PKT) { 16911bc5ad16SMarcel Holtmann err = -EINVAL; 16921bc5ad16SMarcel Holtmann goto drop; 16931bc5ad16SMarcel Holtmann } 16941bc5ad16SMarcel Holtmann 16951bc5ad16SMarcel Holtmann skb_queue_tail(&hdev->raw_q, skb); 16961bc5ad16SMarcel Holtmann queue_work(hdev->workqueue, &hdev->tx_work); 1697d79f34e3SMarcel Holtmann } else if (hci_skb_pkt_type(skb) == HCI_COMMAND_PKT) { 169883985319SHarvey Harrison u16 opcode = get_unaligned_le16(skb->data); 16991da177e4SLinus Torvalds u16 ogf = hci_opcode_ogf(opcode); 17001da177e4SLinus Torvalds u16 ocf = hci_opcode_ocf(opcode); 17011da177e4SLinus Torvalds 17021da177e4SLinus Torvalds if (((ogf > HCI_SFLT_MAX_OGF) || 17033bb3c755SGustavo Padovan !hci_test_bit(ocf & HCI_FLT_OCF_BITS, 17043bb3c755SGustavo Padovan &hci_sec_filter.ocf_mask[ogf])) && 17051da177e4SLinus Torvalds !capable(CAP_NET_RAW)) { 17061da177e4SLinus Torvalds err = -EPERM; 17071da177e4SLinus Torvalds goto drop; 17081da177e4SLinus Torvalds } 17091da177e4SLinus Torvalds 17101982162bSMarcel Holtmann /* Since the opcode has already been extracted here, store 17111982162bSMarcel Holtmann * a copy of the value for later use by the drivers. 17121982162bSMarcel Holtmann */ 17131982162bSMarcel Holtmann hci_skb_opcode(skb) = opcode; 17141982162bSMarcel Holtmann 1715fee746b0SMarcel Holtmann if (ogf == 0x3f) { 17161da177e4SLinus Torvalds skb_queue_tail(&hdev->raw_q, skb); 17173eff45eaSGustavo F. Padovan queue_work(hdev->workqueue, &hdev->tx_work); 17181da177e4SLinus Torvalds } else { 171949c922bbSStephen Hemminger /* Stand-alone HCI commands must be flagged as 172011714b3dSJohan Hedberg * single-command requests. 172111714b3dSJohan Hedberg */ 172244d27137SJohan Hedberg bt_cb(skb)->hci.req_flags |= HCI_REQ_START; 172311714b3dSJohan Hedberg 17241da177e4SLinus Torvalds skb_queue_tail(&hdev->cmd_q, skb); 1725c347b765SGustavo F. Padovan queue_work(hdev->workqueue, &hdev->cmd_work); 17261da177e4SLinus Torvalds } 17271da177e4SLinus Torvalds } else { 17281da177e4SLinus Torvalds if (!capable(CAP_NET_RAW)) { 17291da177e4SLinus Torvalds err = -EPERM; 17301da177e4SLinus Torvalds goto drop; 17311da177e4SLinus Torvalds } 17321da177e4SLinus Torvalds 1733d79f34e3SMarcel Holtmann if (hci_skb_pkt_type(skb) != HCI_ACLDATA_PKT && 1734d79f34e3SMarcel Holtmann hci_skb_pkt_type(skb) != HCI_SCODATA_PKT) { 1735bb77543eSMarcel Holtmann err = -EINVAL; 1736bb77543eSMarcel Holtmann goto drop; 1737bb77543eSMarcel Holtmann } 1738bb77543eSMarcel Holtmann 17391da177e4SLinus Torvalds skb_queue_tail(&hdev->raw_q, skb); 17403eff45eaSGustavo F. Padovan queue_work(hdev->workqueue, &hdev->tx_work); 17411da177e4SLinus Torvalds } 17421da177e4SLinus Torvalds 17431da177e4SLinus Torvalds err = len; 17441da177e4SLinus Torvalds 17451da177e4SLinus Torvalds done: 17461da177e4SLinus Torvalds release_sock(sk); 17471da177e4SLinus Torvalds return err; 17481da177e4SLinus Torvalds 17491da177e4SLinus Torvalds drop: 17501da177e4SLinus Torvalds kfree_skb(skb); 17511da177e4SLinus Torvalds goto done; 17521da177e4SLinus Torvalds } 17531da177e4SLinus Torvalds 17548fc9ced3SGustavo Padovan static int hci_sock_setsockopt(struct socket *sock, int level, int optname, 17558fc9ced3SGustavo Padovan char __user *optval, unsigned int len) 17561da177e4SLinus Torvalds { 17571da177e4SLinus Torvalds struct hci_ufilter uf = { .opcode = 0 }; 17581da177e4SLinus Torvalds struct sock *sk = sock->sk; 17591da177e4SLinus Torvalds int err = 0, opt = 0; 17601da177e4SLinus Torvalds 17611da177e4SLinus Torvalds BT_DBG("sk %p, opt %d", sk, optname); 17621da177e4SLinus Torvalds 176347b0f573SMarcel Holtmann if (level != SOL_HCI) 176447b0f573SMarcel Holtmann return -ENOPROTOOPT; 176547b0f573SMarcel Holtmann 17661da177e4SLinus Torvalds lock_sock(sk); 17671da177e4SLinus Torvalds 17682f39cdb7SMarcel Holtmann if (hci_pi(sk)->channel != HCI_CHANNEL_RAW) { 1769c2371e80SMarcel Holtmann err = -EBADFD; 17702f39cdb7SMarcel Holtmann goto done; 17712f39cdb7SMarcel Holtmann } 17722f39cdb7SMarcel Holtmann 17731da177e4SLinus Torvalds switch (optname) { 17741da177e4SLinus Torvalds case HCI_DATA_DIR: 17751da177e4SLinus Torvalds if (get_user(opt, (int __user *)optval)) { 17761da177e4SLinus Torvalds err = -EFAULT; 17771da177e4SLinus Torvalds break; 17781da177e4SLinus Torvalds } 17791da177e4SLinus Torvalds 17801da177e4SLinus Torvalds if (opt) 17811da177e4SLinus Torvalds hci_pi(sk)->cmsg_mask |= HCI_CMSG_DIR; 17821da177e4SLinus Torvalds else 17831da177e4SLinus Torvalds hci_pi(sk)->cmsg_mask &= ~HCI_CMSG_DIR; 17841da177e4SLinus Torvalds break; 17851da177e4SLinus Torvalds 17861da177e4SLinus Torvalds case HCI_TIME_STAMP: 17871da177e4SLinus Torvalds if (get_user(opt, (int __user *)optval)) { 17881da177e4SLinus Torvalds err = -EFAULT; 17891da177e4SLinus Torvalds break; 17901da177e4SLinus Torvalds } 17911da177e4SLinus Torvalds 17921da177e4SLinus Torvalds if (opt) 17931da177e4SLinus Torvalds hci_pi(sk)->cmsg_mask |= HCI_CMSG_TSTAMP; 17941da177e4SLinus Torvalds else 17951da177e4SLinus Torvalds hci_pi(sk)->cmsg_mask &= ~HCI_CMSG_TSTAMP; 17961da177e4SLinus Torvalds break; 17971da177e4SLinus Torvalds 17981da177e4SLinus Torvalds case HCI_FILTER: 17990878b666SMarcel Holtmann { 18000878b666SMarcel Holtmann struct hci_filter *f = &hci_pi(sk)->filter; 18010878b666SMarcel Holtmann 18020878b666SMarcel Holtmann uf.type_mask = f->type_mask; 18030878b666SMarcel Holtmann uf.opcode = f->opcode; 18040878b666SMarcel Holtmann uf.event_mask[0] = *((u32 *) f->event_mask + 0); 18050878b666SMarcel Holtmann uf.event_mask[1] = *((u32 *) f->event_mask + 1); 18060878b666SMarcel Holtmann } 18070878b666SMarcel Holtmann 18081da177e4SLinus Torvalds len = min_t(unsigned int, len, sizeof(uf)); 18091da177e4SLinus Torvalds if (copy_from_user(&uf, optval, len)) { 18101da177e4SLinus Torvalds err = -EFAULT; 18111da177e4SLinus Torvalds break; 18121da177e4SLinus Torvalds } 18131da177e4SLinus Torvalds 18141da177e4SLinus Torvalds if (!capable(CAP_NET_RAW)) { 18151da177e4SLinus Torvalds uf.type_mask &= hci_sec_filter.type_mask; 18161da177e4SLinus Torvalds uf.event_mask[0] &= *((u32 *) hci_sec_filter.event_mask + 0); 18171da177e4SLinus Torvalds uf.event_mask[1] &= *((u32 *) hci_sec_filter.event_mask + 1); 18181da177e4SLinus Torvalds } 18191da177e4SLinus Torvalds 18201da177e4SLinus Torvalds { 18211da177e4SLinus Torvalds struct hci_filter *f = &hci_pi(sk)->filter; 18221da177e4SLinus Torvalds 18231da177e4SLinus Torvalds f->type_mask = uf.type_mask; 18241da177e4SLinus Torvalds f->opcode = uf.opcode; 18251da177e4SLinus Torvalds *((u32 *) f->event_mask + 0) = uf.event_mask[0]; 18261da177e4SLinus Torvalds *((u32 *) f->event_mask + 1) = uf.event_mask[1]; 18271da177e4SLinus Torvalds } 18281da177e4SLinus Torvalds break; 18291da177e4SLinus Torvalds 18301da177e4SLinus Torvalds default: 18311da177e4SLinus Torvalds err = -ENOPROTOOPT; 18321da177e4SLinus Torvalds break; 18331da177e4SLinus Torvalds } 18341da177e4SLinus Torvalds 18352f39cdb7SMarcel Holtmann done: 18361da177e4SLinus Torvalds release_sock(sk); 18371da177e4SLinus Torvalds return err; 18381da177e4SLinus Torvalds } 18391da177e4SLinus Torvalds 18408fc9ced3SGustavo Padovan static int hci_sock_getsockopt(struct socket *sock, int level, int optname, 18418fc9ced3SGustavo Padovan char __user *optval, int __user *optlen) 18421da177e4SLinus Torvalds { 18431da177e4SLinus Torvalds struct hci_ufilter uf; 18441da177e4SLinus Torvalds struct sock *sk = sock->sk; 1845cedc5469SMarcel Holtmann int len, opt, err = 0; 1846cedc5469SMarcel Holtmann 1847cedc5469SMarcel Holtmann BT_DBG("sk %p, opt %d", sk, optname); 18481da177e4SLinus Torvalds 184947b0f573SMarcel Holtmann if (level != SOL_HCI) 185047b0f573SMarcel Holtmann return -ENOPROTOOPT; 185147b0f573SMarcel Holtmann 18521da177e4SLinus Torvalds if (get_user(len, optlen)) 18531da177e4SLinus Torvalds return -EFAULT; 18541da177e4SLinus Torvalds 1855cedc5469SMarcel Holtmann lock_sock(sk); 1856cedc5469SMarcel Holtmann 1857cedc5469SMarcel Holtmann if (hci_pi(sk)->channel != HCI_CHANNEL_RAW) { 1858c2371e80SMarcel Holtmann err = -EBADFD; 1859cedc5469SMarcel Holtmann goto done; 1860cedc5469SMarcel Holtmann } 1861cedc5469SMarcel Holtmann 18621da177e4SLinus Torvalds switch (optname) { 18631da177e4SLinus Torvalds case HCI_DATA_DIR: 18641da177e4SLinus Torvalds if (hci_pi(sk)->cmsg_mask & HCI_CMSG_DIR) 18651da177e4SLinus Torvalds opt = 1; 18661da177e4SLinus Torvalds else 18671da177e4SLinus Torvalds opt = 0; 18681da177e4SLinus Torvalds 18691da177e4SLinus Torvalds if (put_user(opt, optval)) 1870cedc5469SMarcel Holtmann err = -EFAULT; 18711da177e4SLinus Torvalds break; 18721da177e4SLinus Torvalds 18731da177e4SLinus Torvalds case HCI_TIME_STAMP: 18741da177e4SLinus Torvalds if (hci_pi(sk)->cmsg_mask & HCI_CMSG_TSTAMP) 18751da177e4SLinus Torvalds opt = 1; 18761da177e4SLinus Torvalds else 18771da177e4SLinus Torvalds opt = 0; 18781da177e4SLinus Torvalds 18791da177e4SLinus Torvalds if (put_user(opt, optval)) 1880cedc5469SMarcel Holtmann err = -EFAULT; 18811da177e4SLinus Torvalds break; 18821da177e4SLinus Torvalds 18831da177e4SLinus Torvalds case HCI_FILTER: 18841da177e4SLinus Torvalds { 18851da177e4SLinus Torvalds struct hci_filter *f = &hci_pi(sk)->filter; 18861da177e4SLinus Torvalds 1887e15ca9a0SMathias Krause memset(&uf, 0, sizeof(uf)); 18881da177e4SLinus Torvalds uf.type_mask = f->type_mask; 18891da177e4SLinus Torvalds uf.opcode = f->opcode; 18901da177e4SLinus Torvalds uf.event_mask[0] = *((u32 *) f->event_mask + 0); 18911da177e4SLinus Torvalds uf.event_mask[1] = *((u32 *) f->event_mask + 1); 18921da177e4SLinus Torvalds } 18931da177e4SLinus Torvalds 18941da177e4SLinus Torvalds len = min_t(unsigned int, len, sizeof(uf)); 18951da177e4SLinus Torvalds if (copy_to_user(optval, &uf, len)) 1896cedc5469SMarcel Holtmann err = -EFAULT; 18971da177e4SLinus Torvalds break; 18981da177e4SLinus Torvalds 18991da177e4SLinus Torvalds default: 1900cedc5469SMarcel Holtmann err = -ENOPROTOOPT; 19011da177e4SLinus Torvalds break; 19021da177e4SLinus Torvalds } 19031da177e4SLinus Torvalds 1904cedc5469SMarcel Holtmann done: 1905cedc5469SMarcel Holtmann release_sock(sk); 1906cedc5469SMarcel Holtmann return err; 19071da177e4SLinus Torvalds } 19081da177e4SLinus Torvalds 190990ddc4f0SEric Dumazet static const struct proto_ops hci_sock_ops = { 19101da177e4SLinus Torvalds .family = PF_BLUETOOTH, 19111da177e4SLinus Torvalds .owner = THIS_MODULE, 19121da177e4SLinus Torvalds .release = hci_sock_release, 19131da177e4SLinus Torvalds .bind = hci_sock_bind, 19141da177e4SLinus Torvalds .getname = hci_sock_getname, 19151da177e4SLinus Torvalds .sendmsg = hci_sock_sendmsg, 19161da177e4SLinus Torvalds .recvmsg = hci_sock_recvmsg, 19171da177e4SLinus Torvalds .ioctl = hci_sock_ioctl, 19181da177e4SLinus Torvalds .poll = datagram_poll, 19191da177e4SLinus Torvalds .listen = sock_no_listen, 19201da177e4SLinus Torvalds .shutdown = sock_no_shutdown, 19211da177e4SLinus Torvalds .setsockopt = hci_sock_setsockopt, 19221da177e4SLinus Torvalds .getsockopt = hci_sock_getsockopt, 19231da177e4SLinus Torvalds .connect = sock_no_connect, 19241da177e4SLinus Torvalds .socketpair = sock_no_socketpair, 19251da177e4SLinus Torvalds .accept = sock_no_accept, 19261da177e4SLinus Torvalds .mmap = sock_no_mmap 19271da177e4SLinus Torvalds }; 19281da177e4SLinus Torvalds 19291da177e4SLinus Torvalds static struct proto hci_sk_proto = { 19301da177e4SLinus Torvalds .name = "HCI", 19311da177e4SLinus Torvalds .owner = THIS_MODULE, 19321da177e4SLinus Torvalds .obj_size = sizeof(struct hci_pinfo) 19331da177e4SLinus Torvalds }; 19341da177e4SLinus Torvalds 19353f378b68SEric Paris static int hci_sock_create(struct net *net, struct socket *sock, int protocol, 19363f378b68SEric Paris int kern) 19371da177e4SLinus Torvalds { 19381da177e4SLinus Torvalds struct sock *sk; 19391da177e4SLinus Torvalds 19401da177e4SLinus Torvalds BT_DBG("sock %p", sock); 19411da177e4SLinus Torvalds 19421da177e4SLinus Torvalds if (sock->type != SOCK_RAW) 19431da177e4SLinus Torvalds return -ESOCKTNOSUPPORT; 19441da177e4SLinus Torvalds 19451da177e4SLinus Torvalds sock->ops = &hci_sock_ops; 19461da177e4SLinus Torvalds 194711aa9c28SEric W. Biederman sk = sk_alloc(net, PF_BLUETOOTH, GFP_ATOMIC, &hci_sk_proto, kern); 19481da177e4SLinus Torvalds if (!sk) 19491da177e4SLinus Torvalds return -ENOMEM; 19501da177e4SLinus Torvalds 19511da177e4SLinus Torvalds sock_init_data(sock, sk); 19521da177e4SLinus Torvalds 19531da177e4SLinus Torvalds sock_reset_flag(sk, SOCK_ZAPPED); 19541da177e4SLinus Torvalds 19551da177e4SLinus Torvalds sk->sk_protocol = protocol; 19561da177e4SLinus Torvalds 19571da177e4SLinus Torvalds sock->state = SS_UNCONNECTED; 19581da177e4SLinus Torvalds sk->sk_state = BT_OPEN; 19591da177e4SLinus Torvalds 19601da177e4SLinus Torvalds bt_sock_link(&hci_sk_list, sk); 19611da177e4SLinus Torvalds return 0; 19621da177e4SLinus Torvalds } 19631da177e4SLinus Torvalds 1964ec1b4cf7SStephen Hemminger static const struct net_proto_family hci_sock_family_ops = { 19651da177e4SLinus Torvalds .family = PF_BLUETOOTH, 19661da177e4SLinus Torvalds .owner = THIS_MODULE, 19671da177e4SLinus Torvalds .create = hci_sock_create, 19681da177e4SLinus Torvalds }; 19691da177e4SLinus Torvalds 19701da177e4SLinus Torvalds int __init hci_sock_init(void) 19711da177e4SLinus Torvalds { 19721da177e4SLinus Torvalds int err; 19731da177e4SLinus Torvalds 1974b0a8e282SMarcel Holtmann BUILD_BUG_ON(sizeof(struct sockaddr_hci) > sizeof(struct sockaddr)); 1975b0a8e282SMarcel Holtmann 19761da177e4SLinus Torvalds err = proto_register(&hci_sk_proto, 0); 19771da177e4SLinus Torvalds if (err < 0) 19781da177e4SLinus Torvalds return err; 19791da177e4SLinus Torvalds 19801da177e4SLinus Torvalds err = bt_sock_register(BTPROTO_HCI, &hci_sock_family_ops); 1981f7c86637SMasatake YAMATO if (err < 0) { 1982f7c86637SMasatake YAMATO BT_ERR("HCI socket registration failed"); 19831da177e4SLinus Torvalds goto error; 1984f7c86637SMasatake YAMATO } 1985f7c86637SMasatake YAMATO 1986b0316615SAl Viro err = bt_procfs_init(&init_net, "hci", &hci_sk_list, NULL); 1987f7c86637SMasatake YAMATO if (err < 0) { 1988f7c86637SMasatake YAMATO BT_ERR("Failed to create HCI proc file"); 1989f7c86637SMasatake YAMATO bt_sock_unregister(BTPROTO_HCI); 1990f7c86637SMasatake YAMATO goto error; 1991f7c86637SMasatake YAMATO } 19921da177e4SLinus Torvalds 19931da177e4SLinus Torvalds BT_INFO("HCI socket layer initialized"); 19941da177e4SLinus Torvalds 19951da177e4SLinus Torvalds return 0; 19961da177e4SLinus Torvalds 19971da177e4SLinus Torvalds error: 19981da177e4SLinus Torvalds proto_unregister(&hci_sk_proto); 19991da177e4SLinus Torvalds return err; 20001da177e4SLinus Torvalds } 20011da177e4SLinus Torvalds 2002b7440a14SAnand Gadiyar void hci_sock_cleanup(void) 20031da177e4SLinus Torvalds { 2004f7c86637SMasatake YAMATO bt_procfs_cleanup(&init_net, "hci"); 20055e9d7f86SDavid Herrmann bt_sock_unregister(BTPROTO_HCI); 20061da177e4SLinus Torvalds proto_unregister(&hci_sk_proto); 20071da177e4SLinus Torvalds } 2008