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. */ 267a6038b3SArnd Bergmann #include <linux/compat.h> 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 */ 254a9ee77afSSebastian Andrzej Siewior static 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 261b67bfe0dSSasha Levin sk_for_each(sk, &hci_sk_list.head) { 262470fe1b5SMarcel Holtmann struct sk_buff *nskb; 263470fe1b5SMarcel Holtmann 264c08b1a1dSMarcel Holtmann /* Ignore socket without the flag set */ 265c85be545SMarcel Holtmann if (!hci_sock_test_flag(sk, flag)) 266c08b1a1dSMarcel Holtmann continue; 267c08b1a1dSMarcel Holtmann 268470fe1b5SMarcel Holtmann /* Skip the original socket */ 269470fe1b5SMarcel Holtmann if (sk == skip_sk) 270470fe1b5SMarcel Holtmann continue; 271470fe1b5SMarcel Holtmann 272470fe1b5SMarcel Holtmann if (sk->sk_state != BT_BOUND) 273470fe1b5SMarcel Holtmann continue; 274470fe1b5SMarcel Holtmann 2757129069eSJohan Hedberg if (hci_pi(sk)->channel != channel) 276d7f72f61SMarcel Holtmann continue; 277d7f72f61SMarcel Holtmann 278d7f72f61SMarcel Holtmann nskb = skb_clone(skb, GFP_ATOMIC); 279d7f72f61SMarcel Holtmann if (!nskb) 280d7f72f61SMarcel Holtmann continue; 281d7f72f61SMarcel Holtmann 282d7f72f61SMarcel Holtmann if (sock_queue_rcv_skb(sk, nskb)) 283d7f72f61SMarcel Holtmann kfree_skb(nskb); 284d7f72f61SMarcel Holtmann } 285d7f72f61SMarcel Holtmann 286a9ee77afSSebastian Andrzej Siewior } 287a9ee77afSSebastian Andrzej Siewior 288a9ee77afSSebastian Andrzej Siewior void hci_send_to_channel(unsigned short channel, struct sk_buff *skb, 289a9ee77afSSebastian Andrzej Siewior int flag, struct sock *skip_sk) 290a9ee77afSSebastian Andrzej Siewior { 291a9ee77afSSebastian Andrzej Siewior read_lock(&hci_sk_list.lock); 292a9ee77afSSebastian Andrzej Siewior __hci_send_to_channel(channel, skb, flag, skip_sk); 293d7f72f61SMarcel Holtmann read_unlock(&hci_sk_list.lock); 294d7f72f61SMarcel Holtmann } 295d7f72f61SMarcel Holtmann 296cd82e61cSMarcel Holtmann /* Send frame to monitor socket */ 297cd82e61cSMarcel Holtmann void hci_send_to_monitor(struct hci_dev *hdev, struct sk_buff *skb) 298cd82e61cSMarcel Holtmann { 299cd82e61cSMarcel Holtmann struct sk_buff *skb_copy = NULL; 3002b531294SMarcel Holtmann struct hci_mon_hdr *hdr; 301cd82e61cSMarcel Holtmann __le16 opcode; 302cd82e61cSMarcel Holtmann 303cd82e61cSMarcel Holtmann if (!atomic_read(&monitor_promisc)) 304cd82e61cSMarcel Holtmann return; 305cd82e61cSMarcel Holtmann 306cd82e61cSMarcel Holtmann BT_DBG("hdev %p len %d", hdev, skb->len); 307cd82e61cSMarcel Holtmann 308d79f34e3SMarcel Holtmann switch (hci_skb_pkt_type(skb)) { 309cd82e61cSMarcel Holtmann case HCI_COMMAND_PKT: 310dcf4adbfSJoe Perches opcode = cpu_to_le16(HCI_MON_COMMAND_PKT); 311cd82e61cSMarcel Holtmann break; 312cd82e61cSMarcel Holtmann case HCI_EVENT_PKT: 313dcf4adbfSJoe Perches opcode = cpu_to_le16(HCI_MON_EVENT_PKT); 314cd82e61cSMarcel Holtmann break; 315cd82e61cSMarcel Holtmann case HCI_ACLDATA_PKT: 316cd82e61cSMarcel Holtmann if (bt_cb(skb)->incoming) 317dcf4adbfSJoe Perches opcode = cpu_to_le16(HCI_MON_ACL_RX_PKT); 318cd82e61cSMarcel Holtmann else 319dcf4adbfSJoe Perches opcode = cpu_to_le16(HCI_MON_ACL_TX_PKT); 320cd82e61cSMarcel Holtmann break; 321cd82e61cSMarcel Holtmann case HCI_SCODATA_PKT: 322cd82e61cSMarcel Holtmann if (bt_cb(skb)->incoming) 323dcf4adbfSJoe Perches opcode = cpu_to_le16(HCI_MON_SCO_RX_PKT); 324cd82e61cSMarcel Holtmann else 325dcf4adbfSJoe Perches opcode = cpu_to_le16(HCI_MON_SCO_TX_PKT); 326cd82e61cSMarcel Holtmann break; 327*f9a619dbSLuiz Augusto von Dentz case HCI_ISODATA_PKT: 328*f9a619dbSLuiz Augusto von Dentz if (bt_cb(skb)->incoming) 329*f9a619dbSLuiz Augusto von Dentz opcode = cpu_to_le16(HCI_MON_ISO_RX_PKT); 330*f9a619dbSLuiz Augusto von Dentz else 331*f9a619dbSLuiz Augusto von Dentz opcode = cpu_to_le16(HCI_MON_ISO_TX_PKT); 332*f9a619dbSLuiz Augusto von Dentz break; 333e875ff84SMarcel Holtmann case HCI_DIAG_PKT: 334e875ff84SMarcel Holtmann opcode = cpu_to_le16(HCI_MON_VENDOR_DIAG); 335e875ff84SMarcel Holtmann break; 336cd82e61cSMarcel Holtmann default: 337cd82e61cSMarcel Holtmann return; 338cd82e61cSMarcel Holtmann } 339cd82e61cSMarcel Holtmann 3402b531294SMarcel Holtmann /* Create a private copy with headroom */ 3412b531294SMarcel Holtmann skb_copy = __pskb_copy_fclone(skb, HCI_MON_HDR_SIZE, GFP_ATOMIC, true); 3422b531294SMarcel Holtmann if (!skb_copy) 3432b531294SMarcel Holtmann return; 3442b531294SMarcel Holtmann 3452b531294SMarcel Holtmann /* Put header before the data */ 346d58ff351SJohannes Berg hdr = skb_push(skb_copy, HCI_MON_HDR_SIZE); 3472b531294SMarcel Holtmann hdr->opcode = opcode; 3482b531294SMarcel Holtmann hdr->index = cpu_to_le16(hdev->id); 3492b531294SMarcel Holtmann hdr->len = cpu_to_le16(skb->len); 3502b531294SMarcel Holtmann 351c08b1a1dSMarcel Holtmann hci_send_to_channel(HCI_CHANNEL_MONITOR, skb_copy, 352c08b1a1dSMarcel Holtmann HCI_SOCK_TRUSTED, NULL); 353cd82e61cSMarcel Holtmann kfree_skb(skb_copy); 354cd82e61cSMarcel Holtmann } 355cd82e61cSMarcel Holtmann 35638ceaa00SMarcel Holtmann void hci_send_monitor_ctrl_event(struct hci_dev *hdev, u16 event, 35738ceaa00SMarcel Holtmann void *data, u16 data_len, ktime_t tstamp, 35838ceaa00SMarcel Holtmann int flag, struct sock *skip_sk) 35938ceaa00SMarcel Holtmann { 36038ceaa00SMarcel Holtmann struct sock *sk; 36138ceaa00SMarcel Holtmann __le16 index; 36238ceaa00SMarcel Holtmann 36338ceaa00SMarcel Holtmann if (hdev) 36438ceaa00SMarcel Holtmann index = cpu_to_le16(hdev->id); 36538ceaa00SMarcel Holtmann else 36638ceaa00SMarcel Holtmann index = cpu_to_le16(MGMT_INDEX_NONE); 36738ceaa00SMarcel Holtmann 36838ceaa00SMarcel Holtmann read_lock(&hci_sk_list.lock); 36938ceaa00SMarcel Holtmann 37038ceaa00SMarcel Holtmann sk_for_each(sk, &hci_sk_list.head) { 37138ceaa00SMarcel Holtmann struct hci_mon_hdr *hdr; 37238ceaa00SMarcel Holtmann struct sk_buff *skb; 37338ceaa00SMarcel Holtmann 37438ceaa00SMarcel Holtmann if (hci_pi(sk)->channel != HCI_CHANNEL_CONTROL) 37538ceaa00SMarcel Holtmann continue; 37638ceaa00SMarcel Holtmann 37738ceaa00SMarcel Holtmann /* Ignore socket without the flag set */ 37838ceaa00SMarcel Holtmann if (!hci_sock_test_flag(sk, flag)) 37938ceaa00SMarcel Holtmann continue; 38038ceaa00SMarcel Holtmann 38138ceaa00SMarcel Holtmann /* Skip the original socket */ 38238ceaa00SMarcel Holtmann if (sk == skip_sk) 38338ceaa00SMarcel Holtmann continue; 38438ceaa00SMarcel Holtmann 38538ceaa00SMarcel Holtmann skb = bt_skb_alloc(6 + data_len, GFP_ATOMIC); 38638ceaa00SMarcel Holtmann if (!skb) 38738ceaa00SMarcel Holtmann continue; 38838ceaa00SMarcel Holtmann 38938ceaa00SMarcel Holtmann put_unaligned_le32(hci_pi(sk)->cookie, skb_put(skb, 4)); 39038ceaa00SMarcel Holtmann put_unaligned_le16(event, skb_put(skb, 2)); 39138ceaa00SMarcel Holtmann 39238ceaa00SMarcel Holtmann if (data) 39359ae1d12SJohannes Berg skb_put_data(skb, data, data_len); 39438ceaa00SMarcel Holtmann 39538ceaa00SMarcel Holtmann skb->tstamp = tstamp; 39638ceaa00SMarcel Holtmann 397d58ff351SJohannes Berg hdr = skb_push(skb, HCI_MON_HDR_SIZE); 39838ceaa00SMarcel Holtmann hdr->opcode = cpu_to_le16(HCI_MON_CTRL_EVENT); 39938ceaa00SMarcel Holtmann hdr->index = index; 40038ceaa00SMarcel Holtmann hdr->len = cpu_to_le16(skb->len - HCI_MON_HDR_SIZE); 40138ceaa00SMarcel Holtmann 402a9ee77afSSebastian Andrzej Siewior __hci_send_to_channel(HCI_CHANNEL_MONITOR, skb, 40338ceaa00SMarcel Holtmann HCI_SOCK_TRUSTED, NULL); 40438ceaa00SMarcel Holtmann kfree_skb(skb); 40538ceaa00SMarcel Holtmann } 40638ceaa00SMarcel Holtmann 40738ceaa00SMarcel Holtmann read_unlock(&hci_sk_list.lock); 40838ceaa00SMarcel Holtmann } 40938ceaa00SMarcel Holtmann 410cd82e61cSMarcel Holtmann static struct sk_buff *create_monitor_event(struct hci_dev *hdev, int event) 411cd82e61cSMarcel Holtmann { 412cd82e61cSMarcel Holtmann struct hci_mon_hdr *hdr; 413cd82e61cSMarcel Holtmann struct hci_mon_new_index *ni; 4146c566dd5SMarcel Holtmann struct hci_mon_index_info *ii; 415cd82e61cSMarcel Holtmann struct sk_buff *skb; 416cd82e61cSMarcel Holtmann __le16 opcode; 417cd82e61cSMarcel Holtmann 418cd82e61cSMarcel Holtmann switch (event) { 419cd82e61cSMarcel Holtmann case HCI_DEV_REG: 420cd82e61cSMarcel Holtmann skb = bt_skb_alloc(HCI_MON_NEW_INDEX_SIZE, GFP_ATOMIC); 421cd82e61cSMarcel Holtmann if (!skb) 422cd82e61cSMarcel Holtmann return NULL; 423cd82e61cSMarcel Holtmann 4244df864c1SJohannes Berg ni = skb_put(skb, HCI_MON_NEW_INDEX_SIZE); 425cd82e61cSMarcel Holtmann ni->type = hdev->dev_type; 426cd82e61cSMarcel Holtmann ni->bus = hdev->bus; 427cd82e61cSMarcel Holtmann bacpy(&ni->bdaddr, &hdev->bdaddr); 428cd82e61cSMarcel Holtmann memcpy(ni->name, hdev->name, 8); 429cd82e61cSMarcel Holtmann 430dcf4adbfSJoe Perches opcode = cpu_to_le16(HCI_MON_NEW_INDEX); 431cd82e61cSMarcel Holtmann break; 432cd82e61cSMarcel Holtmann 433cd82e61cSMarcel Holtmann case HCI_DEV_UNREG: 434cd82e61cSMarcel Holtmann skb = bt_skb_alloc(0, GFP_ATOMIC); 435cd82e61cSMarcel Holtmann if (!skb) 436cd82e61cSMarcel Holtmann return NULL; 437cd82e61cSMarcel Holtmann 438dcf4adbfSJoe Perches opcode = cpu_to_le16(HCI_MON_DEL_INDEX); 439cd82e61cSMarcel Holtmann break; 440cd82e61cSMarcel Holtmann 441e131d74aSMarcel Holtmann case HCI_DEV_SETUP: 442e131d74aSMarcel Holtmann if (hdev->manufacturer == 0xffff) 443e131d74aSMarcel Holtmann return NULL; 444e131d74aSMarcel Holtmann 445e131d74aSMarcel Holtmann /* fall through */ 446e131d74aSMarcel Holtmann 4476c566dd5SMarcel Holtmann case HCI_DEV_UP: 4486c566dd5SMarcel Holtmann skb = bt_skb_alloc(HCI_MON_INDEX_INFO_SIZE, GFP_ATOMIC); 4496c566dd5SMarcel Holtmann if (!skb) 4506c566dd5SMarcel Holtmann return NULL; 4516c566dd5SMarcel Holtmann 4524df864c1SJohannes Berg ii = skb_put(skb, HCI_MON_INDEX_INFO_SIZE); 4536c566dd5SMarcel Holtmann bacpy(&ii->bdaddr, &hdev->bdaddr); 4546c566dd5SMarcel Holtmann ii->manufacturer = cpu_to_le16(hdev->manufacturer); 4556c566dd5SMarcel Holtmann 4566c566dd5SMarcel Holtmann opcode = cpu_to_le16(HCI_MON_INDEX_INFO); 4576c566dd5SMarcel Holtmann break; 4586c566dd5SMarcel Holtmann 45922db3cbcSMarcel Holtmann case HCI_DEV_OPEN: 46022db3cbcSMarcel Holtmann skb = bt_skb_alloc(0, GFP_ATOMIC); 46122db3cbcSMarcel Holtmann if (!skb) 46222db3cbcSMarcel Holtmann return NULL; 46322db3cbcSMarcel Holtmann 46422db3cbcSMarcel Holtmann opcode = cpu_to_le16(HCI_MON_OPEN_INDEX); 46522db3cbcSMarcel Holtmann break; 46622db3cbcSMarcel Holtmann 46722db3cbcSMarcel Holtmann case HCI_DEV_CLOSE: 46822db3cbcSMarcel Holtmann skb = bt_skb_alloc(0, GFP_ATOMIC); 46922db3cbcSMarcel Holtmann if (!skb) 47022db3cbcSMarcel Holtmann return NULL; 47122db3cbcSMarcel Holtmann 47222db3cbcSMarcel Holtmann opcode = cpu_to_le16(HCI_MON_CLOSE_INDEX); 47322db3cbcSMarcel Holtmann break; 47422db3cbcSMarcel Holtmann 475cd82e61cSMarcel Holtmann default: 476cd82e61cSMarcel Holtmann return NULL; 477cd82e61cSMarcel Holtmann } 478cd82e61cSMarcel Holtmann 479cd82e61cSMarcel Holtmann __net_timestamp(skb); 480cd82e61cSMarcel Holtmann 481d58ff351SJohannes Berg hdr = skb_push(skb, HCI_MON_HDR_SIZE); 482cd82e61cSMarcel Holtmann hdr->opcode = opcode; 483cd82e61cSMarcel Holtmann hdr->index = cpu_to_le16(hdev->id); 484cd82e61cSMarcel Holtmann hdr->len = cpu_to_le16(skb->len - HCI_MON_HDR_SIZE); 485cd82e61cSMarcel Holtmann 486cd82e61cSMarcel Holtmann return skb; 487cd82e61cSMarcel Holtmann } 488cd82e61cSMarcel Holtmann 489249fa169SMarcel Holtmann static struct sk_buff *create_monitor_ctrl_open(struct sock *sk) 490249fa169SMarcel Holtmann { 491249fa169SMarcel Holtmann struct hci_mon_hdr *hdr; 492249fa169SMarcel Holtmann struct sk_buff *skb; 493d0bef1d2SMarcel Holtmann u16 format; 494249fa169SMarcel Holtmann u8 ver[3]; 495249fa169SMarcel Holtmann u32 flags; 496249fa169SMarcel Holtmann 4970ef2c42fSMarcel Holtmann /* No message needed when cookie is not present */ 4980ef2c42fSMarcel Holtmann if (!hci_pi(sk)->cookie) 4990ef2c42fSMarcel Holtmann return NULL; 5000ef2c42fSMarcel Holtmann 501d0bef1d2SMarcel Holtmann switch (hci_pi(sk)->channel) { 502f81f5b2dSMarcel Holtmann case HCI_CHANNEL_RAW: 503f81f5b2dSMarcel Holtmann format = 0x0000; 504f81f5b2dSMarcel Holtmann ver[0] = BT_SUBSYS_VERSION; 505f81f5b2dSMarcel Holtmann put_unaligned_le16(BT_SUBSYS_REVISION, ver + 1); 506f81f5b2dSMarcel Holtmann break; 507aa1638ddSMarcel Holtmann case HCI_CHANNEL_USER: 508aa1638ddSMarcel Holtmann format = 0x0001; 509aa1638ddSMarcel Holtmann ver[0] = BT_SUBSYS_VERSION; 510aa1638ddSMarcel Holtmann put_unaligned_le16(BT_SUBSYS_REVISION, ver + 1); 511aa1638ddSMarcel Holtmann break; 512d0bef1d2SMarcel Holtmann case HCI_CHANNEL_CONTROL: 513d0bef1d2SMarcel Holtmann format = 0x0002; 514d0bef1d2SMarcel Holtmann mgmt_fill_version_info(ver); 515d0bef1d2SMarcel Holtmann break; 516d0bef1d2SMarcel Holtmann default: 517d0bef1d2SMarcel Holtmann /* No message for unsupported format */ 518d0bef1d2SMarcel Holtmann return NULL; 519d0bef1d2SMarcel Holtmann } 520d0bef1d2SMarcel Holtmann 521249fa169SMarcel Holtmann skb = bt_skb_alloc(14 + TASK_COMM_LEN , GFP_ATOMIC); 522249fa169SMarcel Holtmann if (!skb) 523249fa169SMarcel Holtmann return NULL; 524249fa169SMarcel Holtmann 525249fa169SMarcel Holtmann flags = hci_sock_test_flag(sk, HCI_SOCK_TRUSTED) ? 0x1 : 0x0; 526249fa169SMarcel Holtmann 527249fa169SMarcel Holtmann put_unaligned_le32(hci_pi(sk)->cookie, skb_put(skb, 4)); 528249fa169SMarcel Holtmann put_unaligned_le16(format, skb_put(skb, 2)); 52959ae1d12SJohannes Berg skb_put_data(skb, ver, sizeof(ver)); 530249fa169SMarcel Holtmann put_unaligned_le32(flags, skb_put(skb, 4)); 531634fef61SJohannes Berg skb_put_u8(skb, TASK_COMM_LEN); 53259ae1d12SJohannes Berg skb_put_data(skb, hci_pi(sk)->comm, TASK_COMM_LEN); 533249fa169SMarcel Holtmann 534249fa169SMarcel Holtmann __net_timestamp(skb); 535249fa169SMarcel Holtmann 536d58ff351SJohannes Berg hdr = skb_push(skb, HCI_MON_HDR_SIZE); 537249fa169SMarcel Holtmann hdr->opcode = cpu_to_le16(HCI_MON_CTRL_OPEN); 5380ef2c42fSMarcel Holtmann if (hci_pi(sk)->hdev) 5390ef2c42fSMarcel Holtmann hdr->index = cpu_to_le16(hci_pi(sk)->hdev->id); 5400ef2c42fSMarcel Holtmann else 541249fa169SMarcel Holtmann hdr->index = cpu_to_le16(HCI_DEV_NONE); 542249fa169SMarcel Holtmann hdr->len = cpu_to_le16(skb->len - HCI_MON_HDR_SIZE); 543249fa169SMarcel Holtmann 544249fa169SMarcel Holtmann return skb; 545249fa169SMarcel Holtmann } 546249fa169SMarcel Holtmann 547249fa169SMarcel Holtmann static struct sk_buff *create_monitor_ctrl_close(struct sock *sk) 548249fa169SMarcel Holtmann { 549249fa169SMarcel Holtmann struct hci_mon_hdr *hdr; 550249fa169SMarcel Holtmann struct sk_buff *skb; 551249fa169SMarcel Holtmann 5520ef2c42fSMarcel Holtmann /* No message needed when cookie is not present */ 5530ef2c42fSMarcel Holtmann if (!hci_pi(sk)->cookie) 5540ef2c42fSMarcel Holtmann return NULL; 5550ef2c42fSMarcel Holtmann 556d0bef1d2SMarcel Holtmann switch (hci_pi(sk)->channel) { 557f81f5b2dSMarcel Holtmann case HCI_CHANNEL_RAW: 558aa1638ddSMarcel Holtmann case HCI_CHANNEL_USER: 559d0bef1d2SMarcel Holtmann case HCI_CHANNEL_CONTROL: 560d0bef1d2SMarcel Holtmann break; 561d0bef1d2SMarcel Holtmann default: 562d0bef1d2SMarcel Holtmann /* No message for unsupported format */ 563d0bef1d2SMarcel Holtmann return NULL; 564d0bef1d2SMarcel Holtmann } 565d0bef1d2SMarcel Holtmann 566249fa169SMarcel Holtmann skb = bt_skb_alloc(4, GFP_ATOMIC); 567249fa169SMarcel Holtmann if (!skb) 568249fa169SMarcel Holtmann return NULL; 569249fa169SMarcel Holtmann 570249fa169SMarcel Holtmann put_unaligned_le32(hci_pi(sk)->cookie, skb_put(skb, 4)); 571249fa169SMarcel Holtmann 572249fa169SMarcel Holtmann __net_timestamp(skb); 573249fa169SMarcel Holtmann 574d58ff351SJohannes Berg hdr = skb_push(skb, HCI_MON_HDR_SIZE); 575249fa169SMarcel Holtmann hdr->opcode = cpu_to_le16(HCI_MON_CTRL_CLOSE); 5760ef2c42fSMarcel Holtmann if (hci_pi(sk)->hdev) 5770ef2c42fSMarcel Holtmann hdr->index = cpu_to_le16(hci_pi(sk)->hdev->id); 5780ef2c42fSMarcel Holtmann else 579249fa169SMarcel Holtmann hdr->index = cpu_to_le16(HCI_DEV_NONE); 580249fa169SMarcel Holtmann hdr->len = cpu_to_le16(skb->len - HCI_MON_HDR_SIZE); 581249fa169SMarcel Holtmann 582249fa169SMarcel Holtmann return skb; 583249fa169SMarcel Holtmann } 584249fa169SMarcel Holtmann 58538ceaa00SMarcel Holtmann static struct sk_buff *create_monitor_ctrl_command(struct sock *sk, u16 index, 58638ceaa00SMarcel Holtmann u16 opcode, u16 len, 58738ceaa00SMarcel Holtmann const void *buf) 58838ceaa00SMarcel Holtmann { 58938ceaa00SMarcel Holtmann struct hci_mon_hdr *hdr; 59038ceaa00SMarcel Holtmann struct sk_buff *skb; 59138ceaa00SMarcel Holtmann 59238ceaa00SMarcel Holtmann skb = bt_skb_alloc(6 + len, GFP_ATOMIC); 59338ceaa00SMarcel Holtmann if (!skb) 59438ceaa00SMarcel Holtmann return NULL; 59538ceaa00SMarcel Holtmann 59638ceaa00SMarcel Holtmann put_unaligned_le32(hci_pi(sk)->cookie, skb_put(skb, 4)); 59738ceaa00SMarcel Holtmann put_unaligned_le16(opcode, skb_put(skb, 2)); 59838ceaa00SMarcel Holtmann 59938ceaa00SMarcel Holtmann if (buf) 60059ae1d12SJohannes Berg skb_put_data(skb, buf, len); 60138ceaa00SMarcel Holtmann 60238ceaa00SMarcel Holtmann __net_timestamp(skb); 60338ceaa00SMarcel Holtmann 604d58ff351SJohannes Berg hdr = skb_push(skb, HCI_MON_HDR_SIZE); 60538ceaa00SMarcel Holtmann hdr->opcode = cpu_to_le16(HCI_MON_CTRL_COMMAND); 60638ceaa00SMarcel Holtmann hdr->index = cpu_to_le16(index); 60738ceaa00SMarcel Holtmann hdr->len = cpu_to_le16(skb->len - HCI_MON_HDR_SIZE); 60838ceaa00SMarcel Holtmann 60938ceaa00SMarcel Holtmann return skb; 61038ceaa00SMarcel Holtmann } 61138ceaa00SMarcel Holtmann 612787b306cSJohannes Berg static void __printf(2, 3) 613787b306cSJohannes Berg send_monitor_note(struct sock *sk, const char *fmt, ...) 614dd31506dSMarcel Holtmann { 615787b306cSJohannes Berg size_t len; 616dd31506dSMarcel Holtmann struct hci_mon_hdr *hdr; 617dd31506dSMarcel Holtmann struct sk_buff *skb; 618787b306cSJohannes Berg va_list args; 619787b306cSJohannes Berg 620787b306cSJohannes Berg va_start(args, fmt); 621787b306cSJohannes Berg len = vsnprintf(NULL, 0, fmt, args); 622787b306cSJohannes Berg va_end(args); 623dd31506dSMarcel Holtmann 624dd31506dSMarcel Holtmann skb = bt_skb_alloc(len + 1, GFP_ATOMIC); 625dd31506dSMarcel Holtmann if (!skb) 626dd31506dSMarcel Holtmann return; 627dd31506dSMarcel Holtmann 628787b306cSJohannes Berg va_start(args, fmt); 629787b306cSJohannes Berg vsprintf(skb_put(skb, len), fmt, args); 6304df864c1SJohannes Berg *(u8 *)skb_put(skb, 1) = 0; 631787b306cSJohannes Berg va_end(args); 632dd31506dSMarcel Holtmann 633dd31506dSMarcel Holtmann __net_timestamp(skb); 634dd31506dSMarcel Holtmann 635dd31506dSMarcel Holtmann hdr = (void *)skb_push(skb, HCI_MON_HDR_SIZE); 636dd31506dSMarcel Holtmann hdr->opcode = cpu_to_le16(HCI_MON_SYSTEM_NOTE); 637dd31506dSMarcel Holtmann hdr->index = cpu_to_le16(HCI_DEV_NONE); 638dd31506dSMarcel Holtmann hdr->len = cpu_to_le16(skb->len - HCI_MON_HDR_SIZE); 639dd31506dSMarcel Holtmann 640dd31506dSMarcel Holtmann if (sock_queue_rcv_skb(sk, skb)) 641dd31506dSMarcel Holtmann kfree_skb(skb); 642dd31506dSMarcel Holtmann } 643dd31506dSMarcel Holtmann 644cd82e61cSMarcel Holtmann static void send_monitor_replay(struct sock *sk) 645cd82e61cSMarcel Holtmann { 646cd82e61cSMarcel Holtmann struct hci_dev *hdev; 647cd82e61cSMarcel Holtmann 648cd82e61cSMarcel Holtmann read_lock(&hci_dev_list_lock); 649cd82e61cSMarcel Holtmann 650cd82e61cSMarcel Holtmann list_for_each_entry(hdev, &hci_dev_list, list) { 651cd82e61cSMarcel Holtmann struct sk_buff *skb; 652cd82e61cSMarcel Holtmann 653cd82e61cSMarcel Holtmann skb = create_monitor_event(hdev, HCI_DEV_REG); 654cd82e61cSMarcel Holtmann if (!skb) 655cd82e61cSMarcel Holtmann continue; 656cd82e61cSMarcel Holtmann 657cd82e61cSMarcel Holtmann if (sock_queue_rcv_skb(sk, skb)) 658cd82e61cSMarcel Holtmann kfree_skb(skb); 65922db3cbcSMarcel Holtmann 66022db3cbcSMarcel Holtmann if (!test_bit(HCI_RUNNING, &hdev->flags)) 66122db3cbcSMarcel Holtmann continue; 66222db3cbcSMarcel Holtmann 66322db3cbcSMarcel Holtmann skb = create_monitor_event(hdev, HCI_DEV_OPEN); 66422db3cbcSMarcel Holtmann if (!skb) 66522db3cbcSMarcel Holtmann continue; 66622db3cbcSMarcel Holtmann 66722db3cbcSMarcel Holtmann if (sock_queue_rcv_skb(sk, skb)) 66822db3cbcSMarcel Holtmann kfree_skb(skb); 6696c566dd5SMarcel Holtmann 670e131d74aSMarcel Holtmann if (test_bit(HCI_UP, &hdev->flags)) 6716c566dd5SMarcel Holtmann skb = create_monitor_event(hdev, HCI_DEV_UP); 672e131d74aSMarcel Holtmann else if (hci_dev_test_flag(hdev, HCI_SETUP)) 673e131d74aSMarcel Holtmann skb = create_monitor_event(hdev, HCI_DEV_SETUP); 674e131d74aSMarcel Holtmann else 675e131d74aSMarcel Holtmann skb = NULL; 6766c566dd5SMarcel Holtmann 677e131d74aSMarcel Holtmann if (skb) { 6786c566dd5SMarcel Holtmann if (sock_queue_rcv_skb(sk, skb)) 6796c566dd5SMarcel Holtmann kfree_skb(skb); 680cd82e61cSMarcel Holtmann } 681e131d74aSMarcel Holtmann } 682cd82e61cSMarcel Holtmann 683cd82e61cSMarcel Holtmann read_unlock(&hci_dev_list_lock); 684cd82e61cSMarcel Holtmann } 685cd82e61cSMarcel Holtmann 686249fa169SMarcel Holtmann static void send_monitor_control_replay(struct sock *mon_sk) 687249fa169SMarcel Holtmann { 688249fa169SMarcel Holtmann struct sock *sk; 689249fa169SMarcel Holtmann 690249fa169SMarcel Holtmann read_lock(&hci_sk_list.lock); 691249fa169SMarcel Holtmann 692249fa169SMarcel Holtmann sk_for_each(sk, &hci_sk_list.head) { 693249fa169SMarcel Holtmann struct sk_buff *skb; 694249fa169SMarcel Holtmann 695249fa169SMarcel Holtmann skb = create_monitor_ctrl_open(sk); 696249fa169SMarcel Holtmann if (!skb) 697249fa169SMarcel Holtmann continue; 698249fa169SMarcel Holtmann 699249fa169SMarcel Holtmann if (sock_queue_rcv_skb(mon_sk, skb)) 700249fa169SMarcel Holtmann kfree_skb(skb); 701249fa169SMarcel Holtmann } 702249fa169SMarcel Holtmann 703249fa169SMarcel Holtmann read_unlock(&hci_sk_list.lock); 704249fa169SMarcel Holtmann } 705249fa169SMarcel Holtmann 706040030efSMarcel Holtmann /* Generate internal stack event */ 707040030efSMarcel Holtmann static void hci_si_event(struct hci_dev *hdev, int type, int dlen, void *data) 708040030efSMarcel Holtmann { 709040030efSMarcel Holtmann struct hci_event_hdr *hdr; 710040030efSMarcel Holtmann struct hci_ev_stack_internal *ev; 711040030efSMarcel Holtmann struct sk_buff *skb; 712040030efSMarcel Holtmann 713040030efSMarcel Holtmann skb = bt_skb_alloc(HCI_EVENT_HDR_SIZE + sizeof(*ev) + dlen, GFP_ATOMIC); 714040030efSMarcel Holtmann if (!skb) 715040030efSMarcel Holtmann return; 716040030efSMarcel Holtmann 7174df864c1SJohannes Berg hdr = skb_put(skb, HCI_EVENT_HDR_SIZE); 718040030efSMarcel Holtmann hdr->evt = HCI_EV_STACK_INTERNAL; 719040030efSMarcel Holtmann hdr->plen = sizeof(*ev) + dlen; 720040030efSMarcel Holtmann 7214df864c1SJohannes Berg ev = skb_put(skb, sizeof(*ev) + dlen); 722040030efSMarcel Holtmann ev->type = type; 723040030efSMarcel Holtmann memcpy(ev->data, data, dlen); 724040030efSMarcel Holtmann 725040030efSMarcel Holtmann bt_cb(skb)->incoming = 1; 726040030efSMarcel Holtmann __net_timestamp(skb); 727040030efSMarcel Holtmann 728d79f34e3SMarcel Holtmann hci_skb_pkt_type(skb) = HCI_EVENT_PKT; 729040030efSMarcel Holtmann hci_send_to_sock(hdev, skb); 730040030efSMarcel Holtmann kfree_skb(skb); 731040030efSMarcel Holtmann } 732040030efSMarcel Holtmann 733040030efSMarcel Holtmann void hci_sock_dev_event(struct hci_dev *hdev, int event) 734040030efSMarcel Holtmann { 735040030efSMarcel Holtmann BT_DBG("hdev %s event %d", hdev->name, event); 736040030efSMarcel Holtmann 737cd82e61cSMarcel Holtmann if (atomic_read(&monitor_promisc)) { 738cd82e61cSMarcel Holtmann struct sk_buff *skb; 739cd82e61cSMarcel Holtmann 740ed1b28a4SMarcel Holtmann /* Send event to monitor */ 741cd82e61cSMarcel Holtmann skb = create_monitor_event(hdev, event); 742cd82e61cSMarcel Holtmann if (skb) { 743c08b1a1dSMarcel Holtmann hci_send_to_channel(HCI_CHANNEL_MONITOR, skb, 744c08b1a1dSMarcel Holtmann HCI_SOCK_TRUSTED, NULL); 745cd82e61cSMarcel Holtmann kfree_skb(skb); 746cd82e61cSMarcel Holtmann } 747cd82e61cSMarcel Holtmann } 748cd82e61cSMarcel Holtmann 749ed1b28a4SMarcel Holtmann if (event <= HCI_DEV_DOWN) { 750ed1b28a4SMarcel Holtmann struct hci_ev_si_device ev; 751ed1b28a4SMarcel Holtmann 752040030efSMarcel Holtmann /* Send event to sockets */ 753040030efSMarcel Holtmann ev.event = event; 754040030efSMarcel Holtmann ev.dev_id = hdev->id; 755040030efSMarcel Holtmann hci_si_event(NULL, HCI_EV_SI_DEVICE, sizeof(ev), &ev); 756ed1b28a4SMarcel Holtmann } 757040030efSMarcel Holtmann 758040030efSMarcel Holtmann if (event == HCI_DEV_UNREG) { 759040030efSMarcel Holtmann struct sock *sk; 760040030efSMarcel Holtmann 761040030efSMarcel Holtmann /* Detach sockets from device */ 762040030efSMarcel Holtmann read_lock(&hci_sk_list.lock); 763b67bfe0dSSasha Levin sk_for_each(sk, &hci_sk_list.head) { 764040030efSMarcel Holtmann bh_lock_sock_nested(sk); 765040030efSMarcel Holtmann if (hci_pi(sk)->hdev == hdev) { 766040030efSMarcel Holtmann hci_pi(sk)->hdev = NULL; 767040030efSMarcel Holtmann sk->sk_err = EPIPE; 768040030efSMarcel Holtmann sk->sk_state = BT_OPEN; 769040030efSMarcel Holtmann sk->sk_state_change(sk); 770040030efSMarcel Holtmann 771040030efSMarcel Holtmann hci_dev_put(hdev); 772040030efSMarcel Holtmann } 773040030efSMarcel Holtmann bh_unlock_sock(sk); 774040030efSMarcel Holtmann } 775040030efSMarcel Holtmann read_unlock(&hci_sk_list.lock); 776040030efSMarcel Holtmann } 777040030efSMarcel Holtmann } 778040030efSMarcel Holtmann 779801c1e8dSJohan Hedberg static struct hci_mgmt_chan *__hci_mgmt_chan_find(unsigned short channel) 780801c1e8dSJohan Hedberg { 781801c1e8dSJohan Hedberg struct hci_mgmt_chan *c; 782801c1e8dSJohan Hedberg 783801c1e8dSJohan Hedberg list_for_each_entry(c, &mgmt_chan_list, list) { 784801c1e8dSJohan Hedberg if (c->channel == channel) 785801c1e8dSJohan Hedberg return c; 786801c1e8dSJohan Hedberg } 787801c1e8dSJohan Hedberg 788801c1e8dSJohan Hedberg return NULL; 789801c1e8dSJohan Hedberg } 790801c1e8dSJohan Hedberg 791801c1e8dSJohan Hedberg static struct hci_mgmt_chan *hci_mgmt_chan_find(unsigned short channel) 792801c1e8dSJohan Hedberg { 793801c1e8dSJohan Hedberg struct hci_mgmt_chan *c; 794801c1e8dSJohan Hedberg 795801c1e8dSJohan Hedberg mutex_lock(&mgmt_chan_list_lock); 796801c1e8dSJohan Hedberg c = __hci_mgmt_chan_find(channel); 797801c1e8dSJohan Hedberg mutex_unlock(&mgmt_chan_list_lock); 798801c1e8dSJohan Hedberg 799801c1e8dSJohan Hedberg return c; 800801c1e8dSJohan Hedberg } 801801c1e8dSJohan Hedberg 802801c1e8dSJohan Hedberg int hci_mgmt_chan_register(struct hci_mgmt_chan *c) 803801c1e8dSJohan Hedberg { 804801c1e8dSJohan Hedberg if (c->channel < HCI_CHANNEL_CONTROL) 805801c1e8dSJohan Hedberg return -EINVAL; 806801c1e8dSJohan Hedberg 807801c1e8dSJohan Hedberg mutex_lock(&mgmt_chan_list_lock); 808801c1e8dSJohan Hedberg if (__hci_mgmt_chan_find(c->channel)) { 809801c1e8dSJohan Hedberg mutex_unlock(&mgmt_chan_list_lock); 810801c1e8dSJohan Hedberg return -EALREADY; 811801c1e8dSJohan Hedberg } 812801c1e8dSJohan Hedberg 813801c1e8dSJohan Hedberg list_add_tail(&c->list, &mgmt_chan_list); 814801c1e8dSJohan Hedberg 815801c1e8dSJohan Hedberg mutex_unlock(&mgmt_chan_list_lock); 816801c1e8dSJohan Hedberg 817801c1e8dSJohan Hedberg return 0; 818801c1e8dSJohan Hedberg } 819801c1e8dSJohan Hedberg EXPORT_SYMBOL(hci_mgmt_chan_register); 820801c1e8dSJohan Hedberg 821801c1e8dSJohan Hedberg void hci_mgmt_chan_unregister(struct hci_mgmt_chan *c) 822801c1e8dSJohan Hedberg { 823801c1e8dSJohan Hedberg mutex_lock(&mgmt_chan_list_lock); 824801c1e8dSJohan Hedberg list_del(&c->list); 825801c1e8dSJohan Hedberg mutex_unlock(&mgmt_chan_list_lock); 826801c1e8dSJohan Hedberg } 827801c1e8dSJohan Hedberg EXPORT_SYMBOL(hci_mgmt_chan_unregister); 828801c1e8dSJohan Hedberg 8291da177e4SLinus Torvalds static int hci_sock_release(struct socket *sock) 8301da177e4SLinus Torvalds { 8311da177e4SLinus Torvalds struct sock *sk = sock->sk; 8327b005bd3SMarcel Holtmann struct hci_dev *hdev; 833249fa169SMarcel Holtmann struct sk_buff *skb; 8341da177e4SLinus Torvalds 8351da177e4SLinus Torvalds BT_DBG("sock %p sk %p", sock, sk); 8361da177e4SLinus Torvalds 8371da177e4SLinus Torvalds if (!sk) 8381da177e4SLinus Torvalds return 0; 8391da177e4SLinus Torvalds 84070ecce91SMarcel Holtmann switch (hci_pi(sk)->channel) { 84170ecce91SMarcel Holtmann case HCI_CHANNEL_MONITOR: 842cd82e61cSMarcel Holtmann atomic_dec(&monitor_promisc); 84370ecce91SMarcel Holtmann break; 844f81f5b2dSMarcel Holtmann case HCI_CHANNEL_RAW: 845aa1638ddSMarcel Holtmann case HCI_CHANNEL_USER: 84670ecce91SMarcel Holtmann case HCI_CHANNEL_CONTROL: 847249fa169SMarcel Holtmann /* Send event to monitor */ 848249fa169SMarcel Holtmann skb = create_monitor_ctrl_close(sk); 849249fa169SMarcel Holtmann if (skb) { 850249fa169SMarcel Holtmann hci_send_to_channel(HCI_CHANNEL_MONITOR, skb, 851249fa169SMarcel Holtmann HCI_SOCK_TRUSTED, NULL); 852249fa169SMarcel Holtmann kfree_skb(skb); 853249fa169SMarcel Holtmann } 854249fa169SMarcel Holtmann 855df1cb87aSMarcel Holtmann hci_sock_free_cookie(sk); 85670ecce91SMarcel Holtmann break; 85770ecce91SMarcel Holtmann } 858cd82e61cSMarcel Holtmann 8591da177e4SLinus Torvalds bt_sock_unlink(&hci_sk_list, sk); 8601da177e4SLinus Torvalds 861e20a2e9cSMyungho Jung hdev = hci_pi(sk)->hdev; 8621da177e4SLinus Torvalds if (hdev) { 86323500189SMarcel Holtmann if (hci_pi(sk)->channel == HCI_CHANNEL_USER) { 8649332ef9dSMasahiro Yamada /* When releasing a user channel exclusive access, 8656b3cc1dbSSimon Fels * call hci_dev_do_close directly instead of calling 8666b3cc1dbSSimon Fels * hci_dev_close to ensure the exclusive access will 8676b3cc1dbSSimon Fels * be released and the controller brought back down. 8686b3cc1dbSSimon Fels * 8696b3cc1dbSSimon Fels * The checking of HCI_AUTO_OFF is not needed in this 8706b3cc1dbSSimon Fels * case since it will have been cleared already when 8716b3cc1dbSSimon Fels * opening the user channel. 8726b3cc1dbSSimon Fels */ 8736b3cc1dbSSimon Fels hci_dev_do_close(hdev); 8749380f9eaSLoic Poulain hci_dev_clear_flag(hdev, HCI_USER_CHANNEL); 8759380f9eaSLoic Poulain mgmt_index_added(hdev); 87623500189SMarcel Holtmann } 87723500189SMarcel Holtmann 8781da177e4SLinus Torvalds atomic_dec(&hdev->promisc); 8791da177e4SLinus Torvalds hci_dev_put(hdev); 8801da177e4SLinus Torvalds } 8811da177e4SLinus Torvalds 8821da177e4SLinus Torvalds sock_orphan(sk); 8831da177e4SLinus Torvalds 8841da177e4SLinus Torvalds skb_queue_purge(&sk->sk_receive_queue); 8851da177e4SLinus Torvalds skb_queue_purge(&sk->sk_write_queue); 8861da177e4SLinus Torvalds 8871da177e4SLinus Torvalds sock_put(sk); 8881da177e4SLinus Torvalds return 0; 8891da177e4SLinus Torvalds } 8901da177e4SLinus Torvalds 891b2a66aadSAntti Julku static int hci_sock_blacklist_add(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_add(&hdev->blacklist, &bdaddr, BDADDR_BREDR); 9025e762444SAntti Julku 90309fd0de5SGustavo F. Padovan hci_dev_unlock(hdev); 9045e762444SAntti Julku 9055e762444SAntti Julku return err; 906f0358568SJohan Hedberg } 907f0358568SJohan Hedberg 908b2a66aadSAntti Julku static int hci_sock_blacklist_del(struct hci_dev *hdev, void __user *arg) 909f0358568SJohan Hedberg { 910f0358568SJohan Hedberg bdaddr_t bdaddr; 9115e762444SAntti Julku int err; 912f0358568SJohan Hedberg 913f0358568SJohan Hedberg if (copy_from_user(&bdaddr, arg, sizeof(bdaddr))) 914f0358568SJohan Hedberg return -EFAULT; 915f0358568SJohan Hedberg 91609fd0de5SGustavo F. Padovan hci_dev_lock(hdev); 9175e762444SAntti Julku 918dcc36c16SJohan Hedberg err = hci_bdaddr_list_del(&hdev->blacklist, &bdaddr, BDADDR_BREDR); 9195e762444SAntti Julku 92009fd0de5SGustavo F. Padovan hci_dev_unlock(hdev); 9215e762444SAntti Julku 9225e762444SAntti Julku return err; 923f0358568SJohan Hedberg } 924f0358568SJohan Hedberg 9251da177e4SLinus Torvalds /* Ioctls that require bound socket */ 9266039aa73SGustavo Padovan static int hci_sock_bound_ioctl(struct sock *sk, unsigned int cmd, 9276039aa73SGustavo Padovan unsigned long arg) 9281da177e4SLinus Torvalds { 9291da177e4SLinus Torvalds struct hci_dev *hdev = hci_pi(sk)->hdev; 9301da177e4SLinus Torvalds 9311da177e4SLinus Torvalds if (!hdev) 9321da177e4SLinus Torvalds return -EBADFD; 9331da177e4SLinus Torvalds 934d7a5a11dSMarcel Holtmann if (hci_dev_test_flag(hdev, HCI_USER_CHANNEL)) 9350736cfa8SMarcel Holtmann return -EBUSY; 9360736cfa8SMarcel Holtmann 937d7a5a11dSMarcel Holtmann if (hci_dev_test_flag(hdev, HCI_UNCONFIGURED)) 938fee746b0SMarcel Holtmann return -EOPNOTSUPP; 939fee746b0SMarcel Holtmann 940ca8bee5dSMarcel Holtmann if (hdev->dev_type != HCI_PRIMARY) 9415b69bef5SMarcel Holtmann return -EOPNOTSUPP; 9425b69bef5SMarcel Holtmann 9431da177e4SLinus Torvalds switch (cmd) { 9441da177e4SLinus Torvalds case HCISETRAW: 9451da177e4SLinus Torvalds if (!capable(CAP_NET_ADMIN)) 946bf5b30b8SZhao Hongjiang return -EPERM; 947db596681SMarcel Holtmann return -EOPNOTSUPP; 9481da177e4SLinus Torvalds 9491da177e4SLinus Torvalds case HCIGETCONNINFO: 9501da177e4SLinus Torvalds return hci_get_conn_info(hdev, (void __user *)arg); 9511da177e4SLinus Torvalds 95240be492fSMarcel Holtmann case HCIGETAUTHINFO: 95340be492fSMarcel Holtmann return hci_get_auth_info(hdev, (void __user *)arg); 95440be492fSMarcel Holtmann 955f0358568SJohan Hedberg case HCIBLOCKADDR: 956f0358568SJohan Hedberg if (!capable(CAP_NET_ADMIN)) 957bf5b30b8SZhao Hongjiang return -EPERM; 958b2a66aadSAntti Julku return hci_sock_blacklist_add(hdev, (void __user *)arg); 959f0358568SJohan Hedberg 960f0358568SJohan Hedberg case HCIUNBLOCKADDR: 961f0358568SJohan Hedberg if (!capable(CAP_NET_ADMIN)) 962bf5b30b8SZhao Hongjiang return -EPERM; 963b2a66aadSAntti Julku return hci_sock_blacklist_del(hdev, (void __user *)arg); 9640736cfa8SMarcel Holtmann } 965f0358568SJohan Hedberg 966324d36edSMarcel Holtmann return -ENOIOCTLCMD; 9671da177e4SLinus Torvalds } 9681da177e4SLinus Torvalds 9698fc9ced3SGustavo Padovan static int hci_sock_ioctl(struct socket *sock, unsigned int cmd, 9708fc9ced3SGustavo Padovan unsigned long arg) 9711da177e4SLinus Torvalds { 9721da177e4SLinus Torvalds void __user *argp = (void __user *)arg; 9730736cfa8SMarcel Holtmann struct sock *sk = sock->sk; 9741da177e4SLinus Torvalds int err; 9751da177e4SLinus Torvalds 9761da177e4SLinus Torvalds BT_DBG("cmd %x arg %lx", cmd, arg); 9771da177e4SLinus Torvalds 978c1c4f956SMarcel Holtmann lock_sock(sk); 979c1c4f956SMarcel Holtmann 980c1c4f956SMarcel Holtmann if (hci_pi(sk)->channel != HCI_CHANNEL_RAW) { 981c1c4f956SMarcel Holtmann err = -EBADFD; 982c1c4f956SMarcel Holtmann goto done; 983c1c4f956SMarcel Holtmann } 984c1c4f956SMarcel Holtmann 985f81f5b2dSMarcel Holtmann /* When calling an ioctl on an unbound raw socket, then ensure 986f81f5b2dSMarcel Holtmann * that the monitor gets informed. Ensure that the resulting event 987f81f5b2dSMarcel Holtmann * is only send once by checking if the cookie exists or not. The 988f81f5b2dSMarcel Holtmann * socket cookie will be only ever generated once for the lifetime 989f81f5b2dSMarcel Holtmann * of a given socket. 990f81f5b2dSMarcel Holtmann */ 991f81f5b2dSMarcel Holtmann if (hci_sock_gen_cookie(sk)) { 992f81f5b2dSMarcel Holtmann struct sk_buff *skb; 993f81f5b2dSMarcel Holtmann 994f81f5b2dSMarcel Holtmann if (capable(CAP_NET_ADMIN)) 995f81f5b2dSMarcel Holtmann hci_sock_set_flag(sk, HCI_SOCK_TRUSTED); 996f81f5b2dSMarcel Holtmann 997f81f5b2dSMarcel Holtmann /* Send event to monitor */ 998f81f5b2dSMarcel Holtmann skb = create_monitor_ctrl_open(sk); 999f81f5b2dSMarcel Holtmann if (skb) { 1000f81f5b2dSMarcel Holtmann hci_send_to_channel(HCI_CHANNEL_MONITOR, skb, 1001f81f5b2dSMarcel Holtmann HCI_SOCK_TRUSTED, NULL); 1002f81f5b2dSMarcel Holtmann kfree_skb(skb); 1003f81f5b2dSMarcel Holtmann } 1004f81f5b2dSMarcel Holtmann } 1005f81f5b2dSMarcel Holtmann 1006c1c4f956SMarcel Holtmann release_sock(sk); 1007c1c4f956SMarcel Holtmann 10081da177e4SLinus Torvalds switch (cmd) { 10091da177e4SLinus Torvalds case HCIGETDEVLIST: 10101da177e4SLinus Torvalds return hci_get_dev_list(argp); 10111da177e4SLinus Torvalds 10121da177e4SLinus Torvalds case HCIGETDEVINFO: 10131da177e4SLinus Torvalds return hci_get_dev_info(argp); 10141da177e4SLinus Torvalds 10151da177e4SLinus Torvalds case HCIGETCONNLIST: 10161da177e4SLinus Torvalds return hci_get_conn_list(argp); 10171da177e4SLinus Torvalds 10181da177e4SLinus Torvalds case HCIDEVUP: 10191da177e4SLinus Torvalds if (!capable(CAP_NET_ADMIN)) 1020bf5b30b8SZhao Hongjiang return -EPERM; 10211da177e4SLinus Torvalds return hci_dev_open(arg); 10221da177e4SLinus Torvalds 10231da177e4SLinus Torvalds case HCIDEVDOWN: 10241da177e4SLinus Torvalds if (!capable(CAP_NET_ADMIN)) 1025bf5b30b8SZhao Hongjiang return -EPERM; 10261da177e4SLinus Torvalds return hci_dev_close(arg); 10271da177e4SLinus Torvalds 10281da177e4SLinus Torvalds case HCIDEVRESET: 10291da177e4SLinus Torvalds if (!capable(CAP_NET_ADMIN)) 1030bf5b30b8SZhao Hongjiang return -EPERM; 10311da177e4SLinus Torvalds return hci_dev_reset(arg); 10321da177e4SLinus Torvalds 10331da177e4SLinus Torvalds case HCIDEVRESTAT: 10341da177e4SLinus Torvalds if (!capable(CAP_NET_ADMIN)) 1035bf5b30b8SZhao Hongjiang return -EPERM; 10361da177e4SLinus Torvalds return hci_dev_reset_stat(arg); 10371da177e4SLinus Torvalds 10381da177e4SLinus Torvalds case HCISETSCAN: 10391da177e4SLinus Torvalds case HCISETAUTH: 10401da177e4SLinus Torvalds case HCISETENCRYPT: 10411da177e4SLinus Torvalds case HCISETPTYPE: 10421da177e4SLinus Torvalds case HCISETLINKPOL: 10431da177e4SLinus Torvalds case HCISETLINKMODE: 10441da177e4SLinus Torvalds case HCISETACLMTU: 10451da177e4SLinus Torvalds case HCISETSCOMTU: 10461da177e4SLinus Torvalds if (!capable(CAP_NET_ADMIN)) 1047bf5b30b8SZhao Hongjiang return -EPERM; 10481da177e4SLinus Torvalds return hci_dev_cmd(cmd, argp); 10491da177e4SLinus Torvalds 10501da177e4SLinus Torvalds case HCIINQUIRY: 10511da177e4SLinus Torvalds return hci_inquiry(argp); 1052c1c4f956SMarcel Holtmann } 10531da177e4SLinus Torvalds 10541da177e4SLinus Torvalds lock_sock(sk); 1055c1c4f956SMarcel Holtmann 10561da177e4SLinus Torvalds err = hci_sock_bound_ioctl(sk, cmd, arg); 1057c1c4f956SMarcel Holtmann 1058c1c4f956SMarcel Holtmann done: 10591da177e4SLinus Torvalds release_sock(sk); 10601da177e4SLinus Torvalds return err; 10611da177e4SLinus Torvalds } 10621da177e4SLinus Torvalds 10637a6038b3SArnd Bergmann #ifdef CONFIG_COMPAT 10647a6038b3SArnd Bergmann static int hci_sock_compat_ioctl(struct socket *sock, unsigned int cmd, 10657a6038b3SArnd Bergmann unsigned long arg) 10667a6038b3SArnd Bergmann { 10677a6038b3SArnd Bergmann switch (cmd) { 10687a6038b3SArnd Bergmann case HCIDEVUP: 10697a6038b3SArnd Bergmann case HCIDEVDOWN: 10707a6038b3SArnd Bergmann case HCIDEVRESET: 10717a6038b3SArnd Bergmann case HCIDEVRESTAT: 10727a6038b3SArnd Bergmann return hci_sock_ioctl(sock, cmd, arg); 10737a6038b3SArnd Bergmann } 10747a6038b3SArnd Bergmann 10757a6038b3SArnd Bergmann return hci_sock_ioctl(sock, cmd, (unsigned long)compat_ptr(arg)); 10767a6038b3SArnd Bergmann } 10777a6038b3SArnd Bergmann #endif 10787a6038b3SArnd Bergmann 10798fc9ced3SGustavo Padovan static int hci_sock_bind(struct socket *sock, struct sockaddr *addr, 10808fc9ced3SGustavo Padovan int addr_len) 10811da177e4SLinus Torvalds { 10820381101fSJohan Hedberg struct sockaddr_hci haddr; 10831da177e4SLinus Torvalds struct sock *sk = sock->sk; 10841da177e4SLinus Torvalds struct hci_dev *hdev = NULL; 1085f4cdbb3fSMarcel Holtmann struct sk_buff *skb; 10860381101fSJohan Hedberg int len, err = 0; 10871da177e4SLinus Torvalds 10881da177e4SLinus Torvalds BT_DBG("sock %p sk %p", sock, sk); 10891da177e4SLinus Torvalds 10900381101fSJohan Hedberg if (!addr) 10910381101fSJohan Hedberg return -EINVAL; 10920381101fSJohan Hedberg 10930381101fSJohan Hedberg memset(&haddr, 0, sizeof(haddr)); 10940381101fSJohan Hedberg len = min_t(unsigned int, sizeof(haddr), addr_len); 10950381101fSJohan Hedberg memcpy(&haddr, addr, len); 10960381101fSJohan Hedberg 10970381101fSJohan Hedberg if (haddr.hci_family != AF_BLUETOOTH) 10980381101fSJohan Hedberg return -EINVAL; 10990381101fSJohan Hedberg 11001da177e4SLinus Torvalds lock_sock(sk); 11011da177e4SLinus Torvalds 11027cc2ade2SMarcel Holtmann if (sk->sk_state == BT_BOUND) { 11037cc2ade2SMarcel Holtmann err = -EALREADY; 11047cc2ade2SMarcel Holtmann goto done; 11057cc2ade2SMarcel Holtmann } 11067cc2ade2SMarcel Holtmann 11077cc2ade2SMarcel Holtmann switch (haddr.hci_channel) { 11087cc2ade2SMarcel Holtmann case HCI_CHANNEL_RAW: 11097cc2ade2SMarcel Holtmann if (hci_pi(sk)->hdev) { 11101da177e4SLinus Torvalds err = -EALREADY; 11111da177e4SLinus Torvalds goto done; 11121da177e4SLinus Torvalds } 11131da177e4SLinus Torvalds 11140381101fSJohan Hedberg if (haddr.hci_dev != HCI_DEV_NONE) { 11150381101fSJohan Hedberg hdev = hci_dev_get(haddr.hci_dev); 111670f23020SAndrei Emeltchenko if (!hdev) { 11171da177e4SLinus Torvalds err = -ENODEV; 11181da177e4SLinus Torvalds goto done; 11191da177e4SLinus Torvalds } 11201da177e4SLinus Torvalds 11211da177e4SLinus Torvalds atomic_inc(&hdev->promisc); 11221da177e4SLinus Torvalds } 11231da177e4SLinus Torvalds 11245a6d2cf5SMarcel Holtmann hci_pi(sk)->channel = haddr.hci_channel; 1125f81f5b2dSMarcel Holtmann 1126f4cdbb3fSMarcel Holtmann if (!hci_sock_gen_cookie(sk)) { 1127f4cdbb3fSMarcel Holtmann /* In the case when a cookie has already been assigned, 1128f4cdbb3fSMarcel Holtmann * then there has been already an ioctl issued against 1129f4cdbb3fSMarcel Holtmann * an unbound socket and with that triggerd an open 1130f4cdbb3fSMarcel Holtmann * notification. Send a close notification first to 1131f4cdbb3fSMarcel Holtmann * allow the state transition to bounded. 1132f81f5b2dSMarcel Holtmann */ 1133f4cdbb3fSMarcel Holtmann skb = create_monitor_ctrl_close(sk); 1134f4cdbb3fSMarcel Holtmann if (skb) { 1135f4cdbb3fSMarcel Holtmann hci_send_to_channel(HCI_CHANNEL_MONITOR, skb, 1136f4cdbb3fSMarcel Holtmann HCI_SOCK_TRUSTED, NULL); 1137f4cdbb3fSMarcel Holtmann kfree_skb(skb); 1138f4cdbb3fSMarcel Holtmann } 1139f4cdbb3fSMarcel Holtmann } 1140f81f5b2dSMarcel Holtmann 1141f81f5b2dSMarcel Holtmann if (capable(CAP_NET_ADMIN)) 1142f81f5b2dSMarcel Holtmann hci_sock_set_flag(sk, HCI_SOCK_TRUSTED); 1143f81f5b2dSMarcel Holtmann 1144f4cdbb3fSMarcel Holtmann hci_pi(sk)->hdev = hdev; 1145f4cdbb3fSMarcel Holtmann 1146f81f5b2dSMarcel Holtmann /* Send event to monitor */ 1147f81f5b2dSMarcel Holtmann skb = create_monitor_ctrl_open(sk); 1148f81f5b2dSMarcel Holtmann if (skb) { 1149f81f5b2dSMarcel Holtmann hci_send_to_channel(HCI_CHANNEL_MONITOR, skb, 1150f81f5b2dSMarcel Holtmann HCI_SOCK_TRUSTED, NULL); 1151f81f5b2dSMarcel Holtmann kfree_skb(skb); 1152f81f5b2dSMarcel Holtmann } 11537cc2ade2SMarcel Holtmann break; 11547cc2ade2SMarcel Holtmann 115523500189SMarcel Holtmann case HCI_CHANNEL_USER: 115623500189SMarcel Holtmann if (hci_pi(sk)->hdev) { 115723500189SMarcel Holtmann err = -EALREADY; 115823500189SMarcel Holtmann goto done; 115923500189SMarcel Holtmann } 116023500189SMarcel Holtmann 116123500189SMarcel Holtmann if (haddr.hci_dev == HCI_DEV_NONE) { 116223500189SMarcel Holtmann err = -EINVAL; 116323500189SMarcel Holtmann goto done; 116423500189SMarcel Holtmann } 116523500189SMarcel Holtmann 116610a8b86fSMarcel Holtmann if (!capable(CAP_NET_ADMIN)) { 116723500189SMarcel Holtmann err = -EPERM; 116823500189SMarcel Holtmann goto done; 116923500189SMarcel Holtmann } 117023500189SMarcel Holtmann 117123500189SMarcel Holtmann hdev = hci_dev_get(haddr.hci_dev); 117223500189SMarcel Holtmann if (!hdev) { 117323500189SMarcel Holtmann err = -ENODEV; 117423500189SMarcel Holtmann goto done; 117523500189SMarcel Holtmann } 117623500189SMarcel Holtmann 1177781f899fSMarcel Holtmann if (test_bit(HCI_INIT, &hdev->flags) || 1178d7a5a11dSMarcel Holtmann hci_dev_test_flag(hdev, HCI_SETUP) || 1179781f899fSMarcel Holtmann hci_dev_test_flag(hdev, HCI_CONFIG) || 1180781f899fSMarcel Holtmann (!hci_dev_test_flag(hdev, HCI_AUTO_OFF) && 1181781f899fSMarcel Holtmann test_bit(HCI_UP, &hdev->flags))) { 118223500189SMarcel Holtmann err = -EBUSY; 118323500189SMarcel Holtmann hci_dev_put(hdev); 118423500189SMarcel Holtmann goto done; 118523500189SMarcel Holtmann } 118623500189SMarcel Holtmann 1187238be788SMarcel Holtmann if (hci_dev_test_and_set_flag(hdev, HCI_USER_CHANNEL)) { 118823500189SMarcel Holtmann err = -EUSERS; 118923500189SMarcel Holtmann hci_dev_put(hdev); 119023500189SMarcel Holtmann goto done; 119123500189SMarcel Holtmann } 119223500189SMarcel Holtmann 119323500189SMarcel Holtmann mgmt_index_removed(hdev); 119423500189SMarcel Holtmann 119523500189SMarcel Holtmann err = hci_dev_open(hdev->id); 119623500189SMarcel Holtmann if (err) { 1197781f899fSMarcel Holtmann if (err == -EALREADY) { 1198781f899fSMarcel Holtmann /* In case the transport is already up and 1199781f899fSMarcel Holtmann * running, clear the error here. 1200781f899fSMarcel Holtmann * 12019332ef9dSMasahiro Yamada * This can happen when opening a user 1202781f899fSMarcel Holtmann * channel and HCI_AUTO_OFF grace period 1203781f899fSMarcel Holtmann * is still active. 1204781f899fSMarcel Holtmann */ 1205781f899fSMarcel Holtmann err = 0; 1206781f899fSMarcel Holtmann } else { 1207a358dc11SMarcel Holtmann hci_dev_clear_flag(hdev, HCI_USER_CHANNEL); 1208c6521401SMarcel Holtmann mgmt_index_added(hdev); 120923500189SMarcel Holtmann hci_dev_put(hdev); 121023500189SMarcel Holtmann goto done; 121123500189SMarcel Holtmann } 1212781f899fSMarcel Holtmann } 121323500189SMarcel Holtmann 12145a6d2cf5SMarcel Holtmann hci_pi(sk)->channel = haddr.hci_channel; 1215aa1638ddSMarcel Holtmann 1216aa1638ddSMarcel Holtmann if (!hci_sock_gen_cookie(sk)) { 1217aa1638ddSMarcel Holtmann /* In the case when a cookie has already been assigned, 1218aa1638ddSMarcel Holtmann * this socket will transition from a raw socket into 12199332ef9dSMasahiro Yamada * a user channel socket. For a clean transition, send 1220aa1638ddSMarcel Holtmann * the close notification first. 1221aa1638ddSMarcel Holtmann */ 1222aa1638ddSMarcel Holtmann skb = create_monitor_ctrl_close(sk); 1223aa1638ddSMarcel Holtmann if (skb) { 1224aa1638ddSMarcel Holtmann hci_send_to_channel(HCI_CHANNEL_MONITOR, skb, 1225aa1638ddSMarcel Holtmann HCI_SOCK_TRUSTED, NULL); 1226aa1638ddSMarcel Holtmann kfree_skb(skb); 1227aa1638ddSMarcel Holtmann } 1228aa1638ddSMarcel Holtmann } 1229aa1638ddSMarcel Holtmann 1230aa1638ddSMarcel Holtmann /* The user channel is restricted to CAP_NET_ADMIN 1231aa1638ddSMarcel Holtmann * capabilities and with that implicitly trusted. 1232aa1638ddSMarcel Holtmann */ 1233aa1638ddSMarcel Holtmann hci_sock_set_flag(sk, HCI_SOCK_TRUSTED); 1234aa1638ddSMarcel Holtmann 123523500189SMarcel Holtmann hci_pi(sk)->hdev = hdev; 12365a6d2cf5SMarcel Holtmann 1237aa1638ddSMarcel Holtmann /* Send event to monitor */ 1238aa1638ddSMarcel Holtmann skb = create_monitor_ctrl_open(sk); 1239aa1638ddSMarcel Holtmann if (skb) { 1240aa1638ddSMarcel Holtmann hci_send_to_channel(HCI_CHANNEL_MONITOR, skb, 1241aa1638ddSMarcel Holtmann HCI_SOCK_TRUSTED, NULL); 1242aa1638ddSMarcel Holtmann kfree_skb(skb); 1243aa1638ddSMarcel Holtmann } 1244aa1638ddSMarcel Holtmann 12455a6d2cf5SMarcel Holtmann atomic_inc(&hdev->promisc); 124623500189SMarcel Holtmann break; 124723500189SMarcel Holtmann 1248cd82e61cSMarcel Holtmann case HCI_CHANNEL_MONITOR: 1249cd82e61cSMarcel Holtmann if (haddr.hci_dev != HCI_DEV_NONE) { 1250cd82e61cSMarcel Holtmann err = -EINVAL; 1251cd82e61cSMarcel Holtmann goto done; 1252cd82e61cSMarcel Holtmann } 1253cd82e61cSMarcel Holtmann 1254cd82e61cSMarcel Holtmann if (!capable(CAP_NET_RAW)) { 1255cd82e61cSMarcel Holtmann err = -EPERM; 1256cd82e61cSMarcel Holtmann goto done; 1257cd82e61cSMarcel Holtmann } 1258cd82e61cSMarcel Holtmann 12595a6d2cf5SMarcel Holtmann hci_pi(sk)->channel = haddr.hci_channel; 12605a6d2cf5SMarcel Holtmann 126150ebc055SMarcel Holtmann /* The monitor interface is restricted to CAP_NET_RAW 126250ebc055SMarcel Holtmann * capabilities and with that implicitly trusted. 126350ebc055SMarcel Holtmann */ 126450ebc055SMarcel Holtmann hci_sock_set_flag(sk, HCI_SOCK_TRUSTED); 126550ebc055SMarcel Holtmann 1266787b306cSJohannes Berg send_monitor_note(sk, "Linux version %s (%s)", 1267787b306cSJohannes Berg init_utsname()->release, 1268787b306cSJohannes Berg init_utsname()->machine); 12699e8305b3SMarcel Holtmann send_monitor_note(sk, "Bluetooth subsystem version %u.%u", 12709e8305b3SMarcel Holtmann BT_SUBSYS_VERSION, BT_SUBSYS_REVISION); 1271cd82e61cSMarcel Holtmann send_monitor_replay(sk); 1272249fa169SMarcel Holtmann send_monitor_control_replay(sk); 1273cd82e61cSMarcel Holtmann 1274cd82e61cSMarcel Holtmann atomic_inc(&monitor_promisc); 1275cd82e61cSMarcel Holtmann break; 1276cd82e61cSMarcel Holtmann 1277ac714949SMarcel Holtmann case HCI_CHANNEL_LOGGING: 1278ac714949SMarcel Holtmann if (haddr.hci_dev != HCI_DEV_NONE) { 1279ac714949SMarcel Holtmann err = -EINVAL; 1280ac714949SMarcel Holtmann goto done; 1281ac714949SMarcel Holtmann } 1282ac714949SMarcel Holtmann 1283ac714949SMarcel Holtmann if (!capable(CAP_NET_ADMIN)) { 1284ac714949SMarcel Holtmann err = -EPERM; 1285ac714949SMarcel Holtmann goto done; 1286ac714949SMarcel Holtmann } 12875a6d2cf5SMarcel Holtmann 12885a6d2cf5SMarcel Holtmann hci_pi(sk)->channel = haddr.hci_channel; 1289ac714949SMarcel Holtmann break; 1290ac714949SMarcel Holtmann 12917cc2ade2SMarcel Holtmann default: 1292801c1e8dSJohan Hedberg if (!hci_mgmt_chan_find(haddr.hci_channel)) { 12937cc2ade2SMarcel Holtmann err = -EINVAL; 12947cc2ade2SMarcel Holtmann goto done; 12957cc2ade2SMarcel Holtmann } 12967cc2ade2SMarcel Holtmann 1297801c1e8dSJohan Hedberg if (haddr.hci_dev != HCI_DEV_NONE) { 1298801c1e8dSJohan Hedberg err = -EINVAL; 1299801c1e8dSJohan Hedberg goto done; 1300801c1e8dSJohan Hedberg } 1301801c1e8dSJohan Hedberg 13021195fbb8SMarcel Holtmann /* Users with CAP_NET_ADMIN capabilities are allowed 13031195fbb8SMarcel Holtmann * access to all management commands and events. For 13041195fbb8SMarcel Holtmann * untrusted users the interface is restricted and 13051195fbb8SMarcel Holtmann * also only untrusted events are sent. 130650ebc055SMarcel Holtmann */ 13071195fbb8SMarcel Holtmann if (capable(CAP_NET_ADMIN)) 130850ebc055SMarcel Holtmann hci_sock_set_flag(sk, HCI_SOCK_TRUSTED); 130950ebc055SMarcel Holtmann 13105a6d2cf5SMarcel Holtmann hci_pi(sk)->channel = haddr.hci_channel; 13115a6d2cf5SMarcel Holtmann 1312f9207338SMarcel Holtmann /* At the moment the index and unconfigured index events 1313f9207338SMarcel Holtmann * are enabled unconditionally. Setting them on each 1314f9207338SMarcel Holtmann * socket when binding keeps this functionality. They 1315f9207338SMarcel Holtmann * however might be cleared later and then sending of these 1316f9207338SMarcel Holtmann * events will be disabled, but that is then intentional. 1317f6b7712eSMarcel Holtmann * 1318f6b7712eSMarcel Holtmann * This also enables generic events that are safe to be 1319f6b7712eSMarcel Holtmann * received by untrusted users. Example for such events 1320f6b7712eSMarcel Holtmann * are changes to settings, class of device, name etc. 1321f9207338SMarcel Holtmann */ 13225a6d2cf5SMarcel Holtmann if (hci_pi(sk)->channel == HCI_CHANNEL_CONTROL) { 1323f4cdbb3fSMarcel Holtmann if (!hci_sock_gen_cookie(sk)) { 1324f4cdbb3fSMarcel Holtmann /* In the case when a cookie has already been 1325f4cdbb3fSMarcel Holtmann * assigned, this socket will transtion from 1326f4cdbb3fSMarcel Holtmann * a raw socket into a control socket. To 1327f4cdbb3fSMarcel Holtmann * allow for a clean transtion, send the 1328f4cdbb3fSMarcel Holtmann * close notification first. 1329f4cdbb3fSMarcel Holtmann */ 1330f4cdbb3fSMarcel Holtmann skb = create_monitor_ctrl_close(sk); 1331f4cdbb3fSMarcel Holtmann if (skb) { 1332f4cdbb3fSMarcel Holtmann hci_send_to_channel(HCI_CHANNEL_MONITOR, skb, 1333f4cdbb3fSMarcel Holtmann HCI_SOCK_TRUSTED, NULL); 1334f4cdbb3fSMarcel Holtmann kfree_skb(skb); 1335f4cdbb3fSMarcel Holtmann } 1336f4cdbb3fSMarcel Holtmann } 133770ecce91SMarcel Holtmann 1338249fa169SMarcel Holtmann /* Send event to monitor */ 1339249fa169SMarcel Holtmann skb = create_monitor_ctrl_open(sk); 1340249fa169SMarcel Holtmann if (skb) { 1341249fa169SMarcel Holtmann hci_send_to_channel(HCI_CHANNEL_MONITOR, skb, 1342249fa169SMarcel Holtmann HCI_SOCK_TRUSTED, NULL); 1343249fa169SMarcel Holtmann kfree_skb(skb); 1344249fa169SMarcel Holtmann } 1345249fa169SMarcel Holtmann 1346f9207338SMarcel Holtmann hci_sock_set_flag(sk, HCI_MGMT_INDEX_EVENTS); 1347f9207338SMarcel Holtmann hci_sock_set_flag(sk, HCI_MGMT_UNCONF_INDEX_EVENTS); 13485504c3a3SMarcel Holtmann hci_sock_set_flag(sk, HCI_MGMT_OPTION_EVENTS); 13495504c3a3SMarcel Holtmann hci_sock_set_flag(sk, HCI_MGMT_SETTING_EVENTS); 13505504c3a3SMarcel Holtmann hci_sock_set_flag(sk, HCI_MGMT_DEV_CLASS_EVENTS); 13515504c3a3SMarcel Holtmann hci_sock_set_flag(sk, HCI_MGMT_LOCAL_NAME_EVENTS); 1352f9207338SMarcel Holtmann } 1353801c1e8dSJohan Hedberg break; 1354801c1e8dSJohan Hedberg } 1355801c1e8dSJohan Hedberg 13561da177e4SLinus Torvalds sk->sk_state = BT_BOUND; 13571da177e4SLinus Torvalds 13581da177e4SLinus Torvalds done: 13591da177e4SLinus Torvalds release_sock(sk); 13601da177e4SLinus Torvalds return err; 13611da177e4SLinus Torvalds } 13621da177e4SLinus Torvalds 13638fc9ced3SGustavo Padovan static int hci_sock_getname(struct socket *sock, struct sockaddr *addr, 13649b2c45d4SDenys Vlasenko int peer) 13651da177e4SLinus Torvalds { 13661da177e4SLinus Torvalds struct sockaddr_hci *haddr = (struct sockaddr_hci *)addr; 13671da177e4SLinus Torvalds struct sock *sk = sock->sk; 13689d4b68b2SMarcel Holtmann struct hci_dev *hdev; 13699d4b68b2SMarcel Holtmann int err = 0; 13701da177e4SLinus Torvalds 13711da177e4SLinus Torvalds BT_DBG("sock %p sk %p", sock, sk); 13721da177e4SLinus Torvalds 137306f43cbcSMarcel Holtmann if (peer) 137406f43cbcSMarcel Holtmann return -EOPNOTSUPP; 137506f43cbcSMarcel Holtmann 13761da177e4SLinus Torvalds lock_sock(sk); 13771da177e4SLinus Torvalds 13789d4b68b2SMarcel Holtmann hdev = hci_pi(sk)->hdev; 13799d4b68b2SMarcel Holtmann if (!hdev) { 13809d4b68b2SMarcel Holtmann err = -EBADFD; 13819d4b68b2SMarcel Holtmann goto done; 13829d4b68b2SMarcel Holtmann } 13839d4b68b2SMarcel Holtmann 13841da177e4SLinus Torvalds haddr->hci_family = AF_BLUETOOTH; 13857b005bd3SMarcel Holtmann haddr->hci_dev = hdev->id; 13869d4b68b2SMarcel Holtmann haddr->hci_channel= hci_pi(sk)->channel; 13879b2c45d4SDenys Vlasenko err = sizeof(*haddr); 13881da177e4SLinus Torvalds 13899d4b68b2SMarcel Holtmann done: 13901da177e4SLinus Torvalds release_sock(sk); 13919d4b68b2SMarcel Holtmann return err; 13921da177e4SLinus Torvalds } 13931da177e4SLinus Torvalds 13946039aa73SGustavo Padovan static void hci_sock_cmsg(struct sock *sk, struct msghdr *msg, 13956039aa73SGustavo Padovan struct sk_buff *skb) 13961da177e4SLinus Torvalds { 13971da177e4SLinus Torvalds __u32 mask = hci_pi(sk)->cmsg_mask; 13981da177e4SLinus Torvalds 13990d48d939SMarcel Holtmann if (mask & HCI_CMSG_DIR) { 14000d48d939SMarcel Holtmann int incoming = bt_cb(skb)->incoming; 14018fc9ced3SGustavo Padovan put_cmsg(msg, SOL_HCI, HCI_CMSG_DIR, sizeof(incoming), 14028fc9ced3SGustavo Padovan &incoming); 14030d48d939SMarcel Holtmann } 14041da177e4SLinus Torvalds 1405a61bbcf2SPatrick McHardy if (mask & HCI_CMSG_TSTAMP) { 1406f6e623a6SJohann Felix Soden #ifdef CONFIG_COMPAT 140713c6ee2aSDeepa Dinamani struct old_timeval32 ctv; 1408f6e623a6SJohann Felix Soden #endif 140913c6ee2aSDeepa Dinamani struct __kernel_old_timeval tv; 1410767c5eb5SMarcel Holtmann void *data; 1411767c5eb5SMarcel Holtmann int len; 1412a61bbcf2SPatrick McHardy 1413a61bbcf2SPatrick McHardy skb_get_timestamp(skb, &tv); 1414767c5eb5SMarcel Holtmann 14151da97f83SDavid S. Miller data = &tv; 14161da97f83SDavid S. Miller len = sizeof(tv); 14171da97f83SDavid S. Miller #ifdef CONFIG_COMPAT 1418da88cea1SH. J. Lu if (!COMPAT_USE_64BIT_TIME && 1419da88cea1SH. J. Lu (msg->msg_flags & MSG_CMSG_COMPAT)) { 1420767c5eb5SMarcel Holtmann ctv.tv_sec = tv.tv_sec; 1421767c5eb5SMarcel Holtmann ctv.tv_usec = tv.tv_usec; 1422767c5eb5SMarcel Holtmann data = &ctv; 1423767c5eb5SMarcel Holtmann len = sizeof(ctv); 1424767c5eb5SMarcel Holtmann } 14251da97f83SDavid S. Miller #endif 1426767c5eb5SMarcel Holtmann 1427767c5eb5SMarcel Holtmann put_cmsg(msg, SOL_HCI, HCI_CMSG_TSTAMP, len, data); 1428a61bbcf2SPatrick McHardy } 14291da177e4SLinus Torvalds } 14301da177e4SLinus Torvalds 14318528d3f7SMarcel Holtmann static int hci_sock_recvmsg(struct socket *sock, struct msghdr *msg, 14328528d3f7SMarcel Holtmann size_t len, int flags) 14331da177e4SLinus Torvalds { 14341da177e4SLinus Torvalds int noblock = flags & MSG_DONTWAIT; 14351da177e4SLinus Torvalds struct sock *sk = sock->sk; 14361da177e4SLinus Torvalds struct sk_buff *skb; 14371da177e4SLinus Torvalds int copied, err; 143883871f8cSDenis Kenzior unsigned int skblen; 14391da177e4SLinus Torvalds 14401da177e4SLinus Torvalds BT_DBG("sock %p, sk %p", sock, sk); 14411da177e4SLinus Torvalds 1442d94a6104SMarcel Holtmann if (flags & MSG_OOB) 14431da177e4SLinus Torvalds return -EOPNOTSUPP; 14441da177e4SLinus Torvalds 1445ac714949SMarcel Holtmann if (hci_pi(sk)->channel == HCI_CHANNEL_LOGGING) 1446ac714949SMarcel Holtmann return -EOPNOTSUPP; 1447ac714949SMarcel Holtmann 14481da177e4SLinus Torvalds if (sk->sk_state == BT_CLOSED) 14491da177e4SLinus Torvalds return 0; 14501da177e4SLinus Torvalds 145170f23020SAndrei Emeltchenko skb = skb_recv_datagram(sk, flags, noblock, &err); 145270f23020SAndrei Emeltchenko if (!skb) 14531da177e4SLinus Torvalds return err; 14541da177e4SLinus Torvalds 145583871f8cSDenis Kenzior skblen = skb->len; 14561da177e4SLinus Torvalds copied = skb->len; 14571da177e4SLinus Torvalds if (len < copied) { 14581da177e4SLinus Torvalds msg->msg_flags |= MSG_TRUNC; 14591da177e4SLinus Torvalds copied = len; 14601da177e4SLinus Torvalds } 14611da177e4SLinus Torvalds 1462badff6d0SArnaldo Carvalho de Melo skb_reset_transport_header(skb); 146351f3d02bSDavid S. Miller err = skb_copy_datagram_msg(skb, 0, msg, copied); 14641da177e4SLinus Torvalds 14653a208627SMarcel Holtmann switch (hci_pi(sk)->channel) { 14663a208627SMarcel Holtmann case HCI_CHANNEL_RAW: 14671da177e4SLinus Torvalds hci_sock_cmsg(sk, msg, skb); 14683a208627SMarcel Holtmann break; 146923500189SMarcel Holtmann case HCI_CHANNEL_USER: 1470cd82e61cSMarcel Holtmann case HCI_CHANNEL_MONITOR: 1471cd82e61cSMarcel Holtmann sock_recv_timestamp(msg, sk, skb); 1472cd82e61cSMarcel Holtmann break; 1473801c1e8dSJohan Hedberg default: 1474801c1e8dSJohan Hedberg if (hci_mgmt_chan_find(hci_pi(sk)->channel)) 1475801c1e8dSJohan Hedberg sock_recv_timestamp(msg, sk, skb); 1476801c1e8dSJohan Hedberg break; 14773a208627SMarcel Holtmann } 14781da177e4SLinus Torvalds 14791da177e4SLinus Torvalds skb_free_datagram(sk, skb); 14801da177e4SLinus Torvalds 14814f34228bSLuiz Augusto von Dentz if (flags & MSG_TRUNC) 148283871f8cSDenis Kenzior copied = skblen; 148383871f8cSDenis Kenzior 14841da177e4SLinus Torvalds return err ? : copied; 14851da177e4SLinus Torvalds } 14861da177e4SLinus Torvalds 1487fa4335d7SJohan Hedberg static int hci_mgmt_cmd(struct hci_mgmt_chan *chan, struct sock *sk, 1488fa4335d7SJohan Hedberg struct msghdr *msg, size_t msglen) 1489fa4335d7SJohan Hedberg { 1490fa4335d7SJohan Hedberg void *buf; 1491fa4335d7SJohan Hedberg u8 *cp; 1492fa4335d7SJohan Hedberg struct mgmt_hdr *hdr; 1493fa4335d7SJohan Hedberg u16 opcode, index, len; 1494fa4335d7SJohan Hedberg struct hci_dev *hdev = NULL; 1495fa4335d7SJohan Hedberg const struct hci_mgmt_handler *handler; 1496fa4335d7SJohan Hedberg bool var_len, no_hdev; 1497fa4335d7SJohan Hedberg int err; 1498fa4335d7SJohan Hedberg 1499fa4335d7SJohan Hedberg BT_DBG("got %zu bytes", msglen); 1500fa4335d7SJohan Hedberg 1501fa4335d7SJohan Hedberg if (msglen < sizeof(*hdr)) 1502fa4335d7SJohan Hedberg return -EINVAL; 1503fa4335d7SJohan Hedberg 1504fa4335d7SJohan Hedberg buf = kmalloc(msglen, GFP_KERNEL); 1505fa4335d7SJohan Hedberg if (!buf) 1506fa4335d7SJohan Hedberg return -ENOMEM; 1507fa4335d7SJohan Hedberg 1508fa4335d7SJohan Hedberg if (memcpy_from_msg(buf, msg, msglen)) { 1509fa4335d7SJohan Hedberg err = -EFAULT; 1510fa4335d7SJohan Hedberg goto done; 1511fa4335d7SJohan Hedberg } 1512fa4335d7SJohan Hedberg 1513fa4335d7SJohan Hedberg hdr = buf; 1514fa4335d7SJohan Hedberg opcode = __le16_to_cpu(hdr->opcode); 1515fa4335d7SJohan Hedberg index = __le16_to_cpu(hdr->index); 1516fa4335d7SJohan Hedberg len = __le16_to_cpu(hdr->len); 1517fa4335d7SJohan Hedberg 1518fa4335d7SJohan Hedberg if (len != msglen - sizeof(*hdr)) { 1519fa4335d7SJohan Hedberg err = -EINVAL; 1520fa4335d7SJohan Hedberg goto done; 1521fa4335d7SJohan Hedberg } 1522fa4335d7SJohan Hedberg 152338ceaa00SMarcel Holtmann if (chan->channel == HCI_CHANNEL_CONTROL) { 152438ceaa00SMarcel Holtmann struct sk_buff *skb; 152538ceaa00SMarcel Holtmann 152638ceaa00SMarcel Holtmann /* Send event to monitor */ 152738ceaa00SMarcel Holtmann skb = create_monitor_ctrl_command(sk, index, opcode, len, 152838ceaa00SMarcel Holtmann buf + sizeof(*hdr)); 152938ceaa00SMarcel Holtmann if (skb) { 153038ceaa00SMarcel Holtmann hci_send_to_channel(HCI_CHANNEL_MONITOR, skb, 153138ceaa00SMarcel Holtmann HCI_SOCK_TRUSTED, NULL); 153238ceaa00SMarcel Holtmann kfree_skb(skb); 153338ceaa00SMarcel Holtmann } 153438ceaa00SMarcel Holtmann } 153538ceaa00SMarcel Holtmann 1536fa4335d7SJohan Hedberg if (opcode >= chan->handler_count || 1537fa4335d7SJohan Hedberg chan->handlers[opcode].func == NULL) { 1538fa4335d7SJohan Hedberg BT_DBG("Unknown op %u", opcode); 1539fa4335d7SJohan Hedberg err = mgmt_cmd_status(sk, index, opcode, 1540fa4335d7SJohan Hedberg MGMT_STATUS_UNKNOWN_COMMAND); 1541fa4335d7SJohan Hedberg goto done; 1542fa4335d7SJohan Hedberg } 1543fa4335d7SJohan Hedberg 1544fa4335d7SJohan Hedberg handler = &chan->handlers[opcode]; 1545fa4335d7SJohan Hedberg 1546fa4335d7SJohan Hedberg if (!hci_sock_test_flag(sk, HCI_SOCK_TRUSTED) && 1547fa4335d7SJohan Hedberg !(handler->flags & HCI_MGMT_UNTRUSTED)) { 1548fa4335d7SJohan Hedberg err = mgmt_cmd_status(sk, index, opcode, 1549fa4335d7SJohan Hedberg MGMT_STATUS_PERMISSION_DENIED); 1550fa4335d7SJohan Hedberg goto done; 1551fa4335d7SJohan Hedberg } 1552fa4335d7SJohan Hedberg 1553fa4335d7SJohan Hedberg if (index != MGMT_INDEX_NONE) { 1554fa4335d7SJohan Hedberg hdev = hci_dev_get(index); 1555fa4335d7SJohan Hedberg if (!hdev) { 1556fa4335d7SJohan Hedberg err = mgmt_cmd_status(sk, index, opcode, 1557fa4335d7SJohan Hedberg MGMT_STATUS_INVALID_INDEX); 1558fa4335d7SJohan Hedberg goto done; 1559fa4335d7SJohan Hedberg } 1560fa4335d7SJohan Hedberg 1561fa4335d7SJohan Hedberg if (hci_dev_test_flag(hdev, HCI_SETUP) || 1562fa4335d7SJohan Hedberg hci_dev_test_flag(hdev, HCI_CONFIG) || 1563fa4335d7SJohan Hedberg hci_dev_test_flag(hdev, HCI_USER_CHANNEL)) { 1564fa4335d7SJohan Hedberg err = mgmt_cmd_status(sk, index, opcode, 1565fa4335d7SJohan Hedberg MGMT_STATUS_INVALID_INDEX); 1566fa4335d7SJohan Hedberg goto done; 1567fa4335d7SJohan Hedberg } 1568fa4335d7SJohan Hedberg 1569fa4335d7SJohan Hedberg if (hci_dev_test_flag(hdev, HCI_UNCONFIGURED) && 1570fa4335d7SJohan Hedberg !(handler->flags & HCI_MGMT_UNCONFIGURED)) { 1571fa4335d7SJohan Hedberg err = mgmt_cmd_status(sk, index, opcode, 1572fa4335d7SJohan Hedberg MGMT_STATUS_INVALID_INDEX); 1573fa4335d7SJohan Hedberg goto done; 1574fa4335d7SJohan Hedberg } 1575fa4335d7SJohan Hedberg } 1576fa4335d7SJohan Hedberg 1577fa4335d7SJohan Hedberg no_hdev = (handler->flags & HCI_MGMT_NO_HDEV); 1578fa4335d7SJohan Hedberg if (no_hdev != !hdev) { 1579fa4335d7SJohan Hedberg err = mgmt_cmd_status(sk, index, opcode, 1580fa4335d7SJohan Hedberg MGMT_STATUS_INVALID_INDEX); 1581fa4335d7SJohan Hedberg goto done; 1582fa4335d7SJohan Hedberg } 1583fa4335d7SJohan Hedberg 1584fa4335d7SJohan Hedberg var_len = (handler->flags & HCI_MGMT_VAR_LEN); 1585fa4335d7SJohan Hedberg if ((var_len && len < handler->data_len) || 1586fa4335d7SJohan Hedberg (!var_len && len != handler->data_len)) { 1587fa4335d7SJohan Hedberg err = mgmt_cmd_status(sk, index, opcode, 1588fa4335d7SJohan Hedberg MGMT_STATUS_INVALID_PARAMS); 1589fa4335d7SJohan Hedberg goto done; 1590fa4335d7SJohan Hedberg } 1591fa4335d7SJohan Hedberg 1592fa4335d7SJohan Hedberg if (hdev && chan->hdev_init) 1593fa4335d7SJohan Hedberg chan->hdev_init(sk, hdev); 1594fa4335d7SJohan Hedberg 1595fa4335d7SJohan Hedberg cp = buf + sizeof(*hdr); 1596fa4335d7SJohan Hedberg 1597fa4335d7SJohan Hedberg err = handler->func(sk, hdev, cp, len); 1598fa4335d7SJohan Hedberg if (err < 0) 1599fa4335d7SJohan Hedberg goto done; 1600fa4335d7SJohan Hedberg 1601fa4335d7SJohan Hedberg err = msglen; 1602fa4335d7SJohan Hedberg 1603fa4335d7SJohan Hedberg done: 1604fa4335d7SJohan Hedberg if (hdev) 1605fa4335d7SJohan Hedberg hci_dev_put(hdev); 1606fa4335d7SJohan Hedberg 1607fa4335d7SJohan Hedberg kfree(buf); 1608fa4335d7SJohan Hedberg return err; 1609fa4335d7SJohan Hedberg } 1610fa4335d7SJohan Hedberg 1611ac714949SMarcel Holtmann static int hci_logging_frame(struct sock *sk, struct msghdr *msg, int len) 1612ac714949SMarcel Holtmann { 1613ac714949SMarcel Holtmann struct hci_mon_hdr *hdr; 1614ac714949SMarcel Holtmann struct sk_buff *skb; 1615ac714949SMarcel Holtmann struct hci_dev *hdev; 1616ac714949SMarcel Holtmann u16 index; 1617ac714949SMarcel Holtmann int err; 1618ac714949SMarcel Holtmann 1619ac714949SMarcel Holtmann /* The logging frame consists at minimum of the standard header, 1620ac714949SMarcel Holtmann * the priority byte, the ident length byte and at least one string 1621ac714949SMarcel Holtmann * terminator NUL byte. Anything shorter are invalid packets. 1622ac714949SMarcel Holtmann */ 1623ac714949SMarcel Holtmann if (len < sizeof(*hdr) + 3) 1624ac714949SMarcel Holtmann return -EINVAL; 1625ac714949SMarcel Holtmann 1626ac714949SMarcel Holtmann skb = bt_skb_send_alloc(sk, len, msg->msg_flags & MSG_DONTWAIT, &err); 1627ac714949SMarcel Holtmann if (!skb) 1628ac714949SMarcel Holtmann return err; 1629ac714949SMarcel Holtmann 1630ac714949SMarcel Holtmann if (memcpy_from_msg(skb_put(skb, len), msg, len)) { 1631ac714949SMarcel Holtmann err = -EFAULT; 1632ac714949SMarcel Holtmann goto drop; 1633ac714949SMarcel Holtmann } 1634ac714949SMarcel Holtmann 1635ac714949SMarcel Holtmann hdr = (void *)skb->data; 1636ac714949SMarcel Holtmann 1637ac714949SMarcel Holtmann if (__le16_to_cpu(hdr->len) != len - sizeof(*hdr)) { 1638ac714949SMarcel Holtmann err = -EINVAL; 1639ac714949SMarcel Holtmann goto drop; 1640ac714949SMarcel Holtmann } 1641ac714949SMarcel Holtmann 1642ac714949SMarcel Holtmann if (__le16_to_cpu(hdr->opcode) == 0x0000) { 1643ac714949SMarcel Holtmann __u8 priority = skb->data[sizeof(*hdr)]; 1644ac714949SMarcel Holtmann __u8 ident_len = skb->data[sizeof(*hdr) + 1]; 1645ac714949SMarcel Holtmann 1646ac714949SMarcel Holtmann /* Only the priorities 0-7 are valid and with that any other 1647ac714949SMarcel Holtmann * value results in an invalid packet. 1648ac714949SMarcel Holtmann * 1649ac714949SMarcel Holtmann * The priority byte is followed by an ident length byte and 1650ac714949SMarcel Holtmann * the NUL terminated ident string. Check that the ident 1651ac714949SMarcel Holtmann * length is not overflowing the packet and also that the 1652ac714949SMarcel Holtmann * ident string itself is NUL terminated. In case the ident 1653ac714949SMarcel Holtmann * length is zero, the length value actually doubles as NUL 1654ac714949SMarcel Holtmann * terminator identifier. 1655ac714949SMarcel Holtmann * 1656ac714949SMarcel Holtmann * The message follows the ident string (if present) and 1657ac714949SMarcel Holtmann * must be NUL terminated. Otherwise it is not a valid packet. 1658ac714949SMarcel Holtmann */ 1659ac714949SMarcel Holtmann if (priority > 7 || skb->data[len - 1] != 0x00 || 1660ac714949SMarcel Holtmann ident_len > len - sizeof(*hdr) - 3 || 1661ac714949SMarcel Holtmann skb->data[sizeof(*hdr) + ident_len + 1] != 0x00) { 1662ac714949SMarcel Holtmann err = -EINVAL; 1663ac714949SMarcel Holtmann goto drop; 1664ac714949SMarcel Holtmann } 1665ac714949SMarcel Holtmann } else { 1666ac714949SMarcel Holtmann err = -EINVAL; 1667ac714949SMarcel Holtmann goto drop; 1668ac714949SMarcel Holtmann } 1669ac714949SMarcel Holtmann 1670ac714949SMarcel Holtmann index = __le16_to_cpu(hdr->index); 1671ac714949SMarcel Holtmann 1672ac714949SMarcel Holtmann if (index != MGMT_INDEX_NONE) { 1673ac714949SMarcel Holtmann hdev = hci_dev_get(index); 1674ac714949SMarcel Holtmann if (!hdev) { 1675ac714949SMarcel Holtmann err = -ENODEV; 1676ac714949SMarcel Holtmann goto drop; 1677ac714949SMarcel Holtmann } 1678ac714949SMarcel Holtmann } else { 1679ac714949SMarcel Holtmann hdev = NULL; 1680ac714949SMarcel Holtmann } 1681ac714949SMarcel Holtmann 1682ac714949SMarcel Holtmann hdr->opcode = cpu_to_le16(HCI_MON_USER_LOGGING); 1683ac714949SMarcel Holtmann 1684ac714949SMarcel Holtmann hci_send_to_channel(HCI_CHANNEL_MONITOR, skb, HCI_SOCK_TRUSTED, NULL); 1685ac714949SMarcel Holtmann err = len; 1686ac714949SMarcel Holtmann 1687ac714949SMarcel Holtmann if (hdev) 1688ac714949SMarcel Holtmann hci_dev_put(hdev); 1689ac714949SMarcel Holtmann 1690ac714949SMarcel Holtmann drop: 1691ac714949SMarcel Holtmann kfree_skb(skb); 1692ac714949SMarcel Holtmann return err; 1693ac714949SMarcel Holtmann } 1694ac714949SMarcel Holtmann 16951b784140SYing Xue static int hci_sock_sendmsg(struct socket *sock, struct msghdr *msg, 16961b784140SYing Xue size_t len) 16971da177e4SLinus Torvalds { 16981da177e4SLinus Torvalds struct sock *sk = sock->sk; 1699801c1e8dSJohan Hedberg struct hci_mgmt_chan *chan; 17001da177e4SLinus Torvalds struct hci_dev *hdev; 17011da177e4SLinus Torvalds struct sk_buff *skb; 17021da177e4SLinus Torvalds int err; 17031da177e4SLinus Torvalds 17041da177e4SLinus Torvalds BT_DBG("sock %p sk %p", sock, sk); 17051da177e4SLinus Torvalds 17061da177e4SLinus Torvalds if (msg->msg_flags & MSG_OOB) 17071da177e4SLinus Torvalds return -EOPNOTSUPP; 17081da177e4SLinus Torvalds 1709ab89f0bdSSzymon Janc if (msg->msg_flags & ~(MSG_DONTWAIT|MSG_NOSIGNAL|MSG_ERRQUEUE| 1710ab89f0bdSSzymon Janc MSG_CMSG_COMPAT)) 17111da177e4SLinus Torvalds return -EINVAL; 17121da177e4SLinus Torvalds 17131da177e4SLinus Torvalds if (len < 4 || len > HCI_MAX_FRAME_SIZE) 17141da177e4SLinus Torvalds return -EINVAL; 17151da177e4SLinus Torvalds 17161da177e4SLinus Torvalds lock_sock(sk); 17171da177e4SLinus Torvalds 17180381101fSJohan Hedberg switch (hci_pi(sk)->channel) { 17190381101fSJohan Hedberg case HCI_CHANNEL_RAW: 172023500189SMarcel Holtmann case HCI_CHANNEL_USER: 17210381101fSJohan Hedberg break; 1722cd82e61cSMarcel Holtmann case HCI_CHANNEL_MONITOR: 1723cd82e61cSMarcel Holtmann err = -EOPNOTSUPP; 1724cd82e61cSMarcel Holtmann goto done; 1725ac714949SMarcel Holtmann case HCI_CHANNEL_LOGGING: 1726ac714949SMarcel Holtmann err = hci_logging_frame(sk, msg, len); 1727ac714949SMarcel Holtmann goto done; 17280381101fSJohan Hedberg default: 1729801c1e8dSJohan Hedberg mutex_lock(&mgmt_chan_list_lock); 1730801c1e8dSJohan Hedberg chan = __hci_mgmt_chan_find(hci_pi(sk)->channel); 1731801c1e8dSJohan Hedberg if (chan) 1732fa4335d7SJohan Hedberg err = hci_mgmt_cmd(chan, sk, msg, len); 1733801c1e8dSJohan Hedberg else 17340381101fSJohan Hedberg err = -EINVAL; 1735801c1e8dSJohan Hedberg 1736801c1e8dSJohan Hedberg mutex_unlock(&mgmt_chan_list_lock); 17370381101fSJohan Hedberg goto done; 17380381101fSJohan Hedberg } 17390381101fSJohan Hedberg 174070f23020SAndrei Emeltchenko hdev = hci_pi(sk)->hdev; 174170f23020SAndrei Emeltchenko if (!hdev) { 17421da177e4SLinus Torvalds err = -EBADFD; 17431da177e4SLinus Torvalds goto done; 17441da177e4SLinus Torvalds } 17451da177e4SLinus Torvalds 17467e21addcSMarcel Holtmann if (!test_bit(HCI_UP, &hdev->flags)) { 17477e21addcSMarcel Holtmann err = -ENETDOWN; 17487e21addcSMarcel Holtmann goto done; 17497e21addcSMarcel Holtmann } 17507e21addcSMarcel Holtmann 175170f23020SAndrei Emeltchenko skb = bt_skb_send_alloc(sk, len, msg->msg_flags & MSG_DONTWAIT, &err); 175270f23020SAndrei Emeltchenko if (!skb) 17531da177e4SLinus Torvalds goto done; 17541da177e4SLinus Torvalds 17556ce8e9ceSAl Viro if (memcpy_from_msg(skb_put(skb, len), msg, len)) { 17561da177e4SLinus Torvalds err = -EFAULT; 17571da177e4SLinus Torvalds goto drop; 17581da177e4SLinus Torvalds } 17591da177e4SLinus Torvalds 17608528d3f7SMarcel Holtmann hci_skb_pkt_type(skb) = skb->data[0]; 17611da177e4SLinus Torvalds skb_pull(skb, 1); 17621da177e4SLinus Torvalds 17631bc5ad16SMarcel Holtmann if (hci_pi(sk)->channel == HCI_CHANNEL_USER) { 17641bc5ad16SMarcel Holtmann /* No permission check is needed for user channel 17651bc5ad16SMarcel Holtmann * since that gets enforced when binding the socket. 17661bc5ad16SMarcel Holtmann * 17671bc5ad16SMarcel Holtmann * However check that the packet type is valid. 17681bc5ad16SMarcel Holtmann */ 1769d79f34e3SMarcel Holtmann if (hci_skb_pkt_type(skb) != HCI_COMMAND_PKT && 1770d79f34e3SMarcel Holtmann hci_skb_pkt_type(skb) != HCI_ACLDATA_PKT && 1771d79f34e3SMarcel Holtmann hci_skb_pkt_type(skb) != HCI_SCODATA_PKT) { 17721bc5ad16SMarcel Holtmann err = -EINVAL; 17731bc5ad16SMarcel Holtmann goto drop; 17741bc5ad16SMarcel Holtmann } 17751bc5ad16SMarcel Holtmann 17761bc5ad16SMarcel Holtmann skb_queue_tail(&hdev->raw_q, skb); 17771bc5ad16SMarcel Holtmann queue_work(hdev->workqueue, &hdev->tx_work); 1778d79f34e3SMarcel Holtmann } else if (hci_skb_pkt_type(skb) == HCI_COMMAND_PKT) { 177983985319SHarvey Harrison u16 opcode = get_unaligned_le16(skb->data); 17801da177e4SLinus Torvalds u16 ogf = hci_opcode_ogf(opcode); 17811da177e4SLinus Torvalds u16 ocf = hci_opcode_ocf(opcode); 17821da177e4SLinus Torvalds 17831da177e4SLinus Torvalds if (((ogf > HCI_SFLT_MAX_OGF) || 17843bb3c755SGustavo Padovan !hci_test_bit(ocf & HCI_FLT_OCF_BITS, 17853bb3c755SGustavo Padovan &hci_sec_filter.ocf_mask[ogf])) && 17861da177e4SLinus Torvalds !capable(CAP_NET_RAW)) { 17871da177e4SLinus Torvalds err = -EPERM; 17881da177e4SLinus Torvalds goto drop; 17891da177e4SLinus Torvalds } 17901da177e4SLinus Torvalds 17911982162bSMarcel Holtmann /* Since the opcode has already been extracted here, store 17921982162bSMarcel Holtmann * a copy of the value for later use by the drivers. 17931982162bSMarcel Holtmann */ 17941982162bSMarcel Holtmann hci_skb_opcode(skb) = opcode; 17951982162bSMarcel Holtmann 1796fee746b0SMarcel Holtmann if (ogf == 0x3f) { 17971da177e4SLinus Torvalds skb_queue_tail(&hdev->raw_q, skb); 17983eff45eaSGustavo F. Padovan queue_work(hdev->workqueue, &hdev->tx_work); 17991da177e4SLinus Torvalds } else { 180049c922bbSStephen Hemminger /* Stand-alone HCI commands must be flagged as 180111714b3dSJohan Hedberg * single-command requests. 180211714b3dSJohan Hedberg */ 180344d27137SJohan Hedberg bt_cb(skb)->hci.req_flags |= HCI_REQ_START; 180411714b3dSJohan Hedberg 18051da177e4SLinus Torvalds skb_queue_tail(&hdev->cmd_q, skb); 1806c347b765SGustavo F. Padovan queue_work(hdev->workqueue, &hdev->cmd_work); 18071da177e4SLinus Torvalds } 18081da177e4SLinus Torvalds } else { 18091da177e4SLinus Torvalds if (!capable(CAP_NET_RAW)) { 18101da177e4SLinus Torvalds err = -EPERM; 18111da177e4SLinus Torvalds goto drop; 18121da177e4SLinus Torvalds } 18131da177e4SLinus Torvalds 1814d79f34e3SMarcel Holtmann if (hci_skb_pkt_type(skb) != HCI_ACLDATA_PKT && 1815d79f34e3SMarcel Holtmann hci_skb_pkt_type(skb) != HCI_SCODATA_PKT) { 1816bb77543eSMarcel Holtmann err = -EINVAL; 1817bb77543eSMarcel Holtmann goto drop; 1818bb77543eSMarcel Holtmann } 1819bb77543eSMarcel Holtmann 18201da177e4SLinus Torvalds skb_queue_tail(&hdev->raw_q, skb); 18213eff45eaSGustavo F. Padovan queue_work(hdev->workqueue, &hdev->tx_work); 18221da177e4SLinus Torvalds } 18231da177e4SLinus Torvalds 18241da177e4SLinus Torvalds err = len; 18251da177e4SLinus Torvalds 18261da177e4SLinus Torvalds done: 18271da177e4SLinus Torvalds release_sock(sk); 18281da177e4SLinus Torvalds return err; 18291da177e4SLinus Torvalds 18301da177e4SLinus Torvalds drop: 18311da177e4SLinus Torvalds kfree_skb(skb); 18321da177e4SLinus Torvalds goto done; 18331da177e4SLinus Torvalds } 18341da177e4SLinus Torvalds 18358fc9ced3SGustavo Padovan static int hci_sock_setsockopt(struct socket *sock, int level, int optname, 18368fc9ced3SGustavo Padovan char __user *optval, unsigned int len) 18371da177e4SLinus Torvalds { 18381da177e4SLinus Torvalds struct hci_ufilter uf = { .opcode = 0 }; 18391da177e4SLinus Torvalds struct sock *sk = sock->sk; 18401da177e4SLinus Torvalds int err = 0, opt = 0; 18411da177e4SLinus Torvalds 18421da177e4SLinus Torvalds BT_DBG("sk %p, opt %d", sk, optname); 18431da177e4SLinus Torvalds 184447b0f573SMarcel Holtmann if (level != SOL_HCI) 184547b0f573SMarcel Holtmann return -ENOPROTOOPT; 184647b0f573SMarcel Holtmann 18471da177e4SLinus Torvalds lock_sock(sk); 18481da177e4SLinus Torvalds 18492f39cdb7SMarcel Holtmann if (hci_pi(sk)->channel != HCI_CHANNEL_RAW) { 1850c2371e80SMarcel Holtmann err = -EBADFD; 18512f39cdb7SMarcel Holtmann goto done; 18522f39cdb7SMarcel Holtmann } 18532f39cdb7SMarcel Holtmann 18541da177e4SLinus Torvalds switch (optname) { 18551da177e4SLinus Torvalds case HCI_DATA_DIR: 18561da177e4SLinus Torvalds if (get_user(opt, (int __user *)optval)) { 18571da177e4SLinus Torvalds err = -EFAULT; 18581da177e4SLinus Torvalds break; 18591da177e4SLinus Torvalds } 18601da177e4SLinus Torvalds 18611da177e4SLinus Torvalds if (opt) 18621da177e4SLinus Torvalds hci_pi(sk)->cmsg_mask |= HCI_CMSG_DIR; 18631da177e4SLinus Torvalds else 18641da177e4SLinus Torvalds hci_pi(sk)->cmsg_mask &= ~HCI_CMSG_DIR; 18651da177e4SLinus Torvalds break; 18661da177e4SLinus Torvalds 18671da177e4SLinus Torvalds case HCI_TIME_STAMP: 18681da177e4SLinus Torvalds if (get_user(opt, (int __user *)optval)) { 18691da177e4SLinus Torvalds err = -EFAULT; 18701da177e4SLinus Torvalds break; 18711da177e4SLinus Torvalds } 18721da177e4SLinus Torvalds 18731da177e4SLinus Torvalds if (opt) 18741da177e4SLinus Torvalds hci_pi(sk)->cmsg_mask |= HCI_CMSG_TSTAMP; 18751da177e4SLinus Torvalds else 18761da177e4SLinus Torvalds hci_pi(sk)->cmsg_mask &= ~HCI_CMSG_TSTAMP; 18771da177e4SLinus Torvalds break; 18781da177e4SLinus Torvalds 18791da177e4SLinus Torvalds case HCI_FILTER: 18800878b666SMarcel Holtmann { 18810878b666SMarcel Holtmann struct hci_filter *f = &hci_pi(sk)->filter; 18820878b666SMarcel Holtmann 18830878b666SMarcel Holtmann uf.type_mask = f->type_mask; 18840878b666SMarcel Holtmann uf.opcode = f->opcode; 18850878b666SMarcel Holtmann uf.event_mask[0] = *((u32 *) f->event_mask + 0); 18860878b666SMarcel Holtmann uf.event_mask[1] = *((u32 *) f->event_mask + 1); 18870878b666SMarcel Holtmann } 18880878b666SMarcel Holtmann 18891da177e4SLinus Torvalds len = min_t(unsigned int, len, sizeof(uf)); 18901da177e4SLinus Torvalds if (copy_from_user(&uf, optval, len)) { 18911da177e4SLinus Torvalds err = -EFAULT; 18921da177e4SLinus Torvalds break; 18931da177e4SLinus Torvalds } 18941da177e4SLinus Torvalds 18951da177e4SLinus Torvalds if (!capable(CAP_NET_RAW)) { 18961da177e4SLinus Torvalds uf.type_mask &= hci_sec_filter.type_mask; 18971da177e4SLinus Torvalds uf.event_mask[0] &= *((u32 *) hci_sec_filter.event_mask + 0); 18981da177e4SLinus Torvalds uf.event_mask[1] &= *((u32 *) hci_sec_filter.event_mask + 1); 18991da177e4SLinus Torvalds } 19001da177e4SLinus Torvalds 19011da177e4SLinus Torvalds { 19021da177e4SLinus Torvalds struct hci_filter *f = &hci_pi(sk)->filter; 19031da177e4SLinus Torvalds 19041da177e4SLinus Torvalds f->type_mask = uf.type_mask; 19051da177e4SLinus Torvalds f->opcode = uf.opcode; 19061da177e4SLinus Torvalds *((u32 *) f->event_mask + 0) = uf.event_mask[0]; 19071da177e4SLinus Torvalds *((u32 *) f->event_mask + 1) = uf.event_mask[1]; 19081da177e4SLinus Torvalds } 19091da177e4SLinus Torvalds break; 19101da177e4SLinus Torvalds 19111da177e4SLinus Torvalds default: 19121da177e4SLinus Torvalds err = -ENOPROTOOPT; 19131da177e4SLinus Torvalds break; 19141da177e4SLinus Torvalds } 19151da177e4SLinus Torvalds 19162f39cdb7SMarcel Holtmann done: 19171da177e4SLinus Torvalds release_sock(sk); 19181da177e4SLinus Torvalds return err; 19191da177e4SLinus Torvalds } 19201da177e4SLinus Torvalds 19218fc9ced3SGustavo Padovan static int hci_sock_getsockopt(struct socket *sock, int level, int optname, 19228fc9ced3SGustavo Padovan char __user *optval, int __user *optlen) 19231da177e4SLinus Torvalds { 19241da177e4SLinus Torvalds struct hci_ufilter uf; 19251da177e4SLinus Torvalds struct sock *sk = sock->sk; 1926cedc5469SMarcel Holtmann int len, opt, err = 0; 1927cedc5469SMarcel Holtmann 1928cedc5469SMarcel Holtmann BT_DBG("sk %p, opt %d", sk, optname); 19291da177e4SLinus Torvalds 193047b0f573SMarcel Holtmann if (level != SOL_HCI) 193147b0f573SMarcel Holtmann return -ENOPROTOOPT; 193247b0f573SMarcel Holtmann 19331da177e4SLinus Torvalds if (get_user(len, optlen)) 19341da177e4SLinus Torvalds return -EFAULT; 19351da177e4SLinus Torvalds 1936cedc5469SMarcel Holtmann lock_sock(sk); 1937cedc5469SMarcel Holtmann 1938cedc5469SMarcel Holtmann if (hci_pi(sk)->channel != HCI_CHANNEL_RAW) { 1939c2371e80SMarcel Holtmann err = -EBADFD; 1940cedc5469SMarcel Holtmann goto done; 1941cedc5469SMarcel Holtmann } 1942cedc5469SMarcel Holtmann 19431da177e4SLinus Torvalds switch (optname) { 19441da177e4SLinus Torvalds case HCI_DATA_DIR: 19451da177e4SLinus Torvalds if (hci_pi(sk)->cmsg_mask & HCI_CMSG_DIR) 19461da177e4SLinus Torvalds opt = 1; 19471da177e4SLinus Torvalds else 19481da177e4SLinus Torvalds opt = 0; 19491da177e4SLinus Torvalds 19501da177e4SLinus Torvalds if (put_user(opt, optval)) 1951cedc5469SMarcel Holtmann err = -EFAULT; 19521da177e4SLinus Torvalds break; 19531da177e4SLinus Torvalds 19541da177e4SLinus Torvalds case HCI_TIME_STAMP: 19551da177e4SLinus Torvalds if (hci_pi(sk)->cmsg_mask & HCI_CMSG_TSTAMP) 19561da177e4SLinus Torvalds opt = 1; 19571da177e4SLinus Torvalds else 19581da177e4SLinus Torvalds opt = 0; 19591da177e4SLinus Torvalds 19601da177e4SLinus Torvalds if (put_user(opt, optval)) 1961cedc5469SMarcel Holtmann err = -EFAULT; 19621da177e4SLinus Torvalds break; 19631da177e4SLinus Torvalds 19641da177e4SLinus Torvalds case HCI_FILTER: 19651da177e4SLinus Torvalds { 19661da177e4SLinus Torvalds struct hci_filter *f = &hci_pi(sk)->filter; 19671da177e4SLinus Torvalds 1968e15ca9a0SMathias Krause memset(&uf, 0, sizeof(uf)); 19691da177e4SLinus Torvalds uf.type_mask = f->type_mask; 19701da177e4SLinus Torvalds uf.opcode = f->opcode; 19711da177e4SLinus Torvalds uf.event_mask[0] = *((u32 *) f->event_mask + 0); 19721da177e4SLinus Torvalds uf.event_mask[1] = *((u32 *) f->event_mask + 1); 19731da177e4SLinus Torvalds } 19741da177e4SLinus Torvalds 19751da177e4SLinus Torvalds len = min_t(unsigned int, len, sizeof(uf)); 19761da177e4SLinus Torvalds if (copy_to_user(optval, &uf, len)) 1977cedc5469SMarcel Holtmann err = -EFAULT; 19781da177e4SLinus Torvalds break; 19791da177e4SLinus Torvalds 19801da177e4SLinus Torvalds default: 1981cedc5469SMarcel Holtmann err = -ENOPROTOOPT; 19821da177e4SLinus Torvalds break; 19831da177e4SLinus Torvalds } 19841da177e4SLinus Torvalds 1985cedc5469SMarcel Holtmann done: 1986cedc5469SMarcel Holtmann release_sock(sk); 1987cedc5469SMarcel Holtmann return err; 19881da177e4SLinus Torvalds } 19891da177e4SLinus Torvalds 199090ddc4f0SEric Dumazet static const struct proto_ops hci_sock_ops = { 19911da177e4SLinus Torvalds .family = PF_BLUETOOTH, 19921da177e4SLinus Torvalds .owner = THIS_MODULE, 19931da177e4SLinus Torvalds .release = hci_sock_release, 19941da177e4SLinus Torvalds .bind = hci_sock_bind, 19951da177e4SLinus Torvalds .getname = hci_sock_getname, 19961da177e4SLinus Torvalds .sendmsg = hci_sock_sendmsg, 19971da177e4SLinus Torvalds .recvmsg = hci_sock_recvmsg, 19981da177e4SLinus Torvalds .ioctl = hci_sock_ioctl, 19997a6038b3SArnd Bergmann #ifdef CONFIG_COMPAT 20007a6038b3SArnd Bergmann .compat_ioctl = hci_sock_compat_ioctl, 20017a6038b3SArnd Bergmann #endif 2002a11e1d43SLinus Torvalds .poll = datagram_poll, 20031da177e4SLinus Torvalds .listen = sock_no_listen, 20041da177e4SLinus Torvalds .shutdown = sock_no_shutdown, 20051da177e4SLinus Torvalds .setsockopt = hci_sock_setsockopt, 20061da177e4SLinus Torvalds .getsockopt = hci_sock_getsockopt, 20071da177e4SLinus Torvalds .connect = sock_no_connect, 20081da177e4SLinus Torvalds .socketpair = sock_no_socketpair, 20091da177e4SLinus Torvalds .accept = sock_no_accept, 20101da177e4SLinus Torvalds .mmap = sock_no_mmap 20111da177e4SLinus Torvalds }; 20121da177e4SLinus Torvalds 20131da177e4SLinus Torvalds static struct proto hci_sk_proto = { 20141da177e4SLinus Torvalds .name = "HCI", 20151da177e4SLinus Torvalds .owner = THIS_MODULE, 20161da177e4SLinus Torvalds .obj_size = sizeof(struct hci_pinfo) 20171da177e4SLinus Torvalds }; 20181da177e4SLinus Torvalds 20193f378b68SEric Paris static int hci_sock_create(struct net *net, struct socket *sock, int protocol, 20203f378b68SEric Paris int kern) 20211da177e4SLinus Torvalds { 20221da177e4SLinus Torvalds struct sock *sk; 20231da177e4SLinus Torvalds 20241da177e4SLinus Torvalds BT_DBG("sock %p", sock); 20251da177e4SLinus Torvalds 20261da177e4SLinus Torvalds if (sock->type != SOCK_RAW) 20271da177e4SLinus Torvalds return -ESOCKTNOSUPPORT; 20281da177e4SLinus Torvalds 20291da177e4SLinus Torvalds sock->ops = &hci_sock_ops; 20301da177e4SLinus Torvalds 203111aa9c28SEric W. Biederman sk = sk_alloc(net, PF_BLUETOOTH, GFP_ATOMIC, &hci_sk_proto, kern); 20321da177e4SLinus Torvalds if (!sk) 20331da177e4SLinus Torvalds return -ENOMEM; 20341da177e4SLinus Torvalds 20351da177e4SLinus Torvalds sock_init_data(sock, sk); 20361da177e4SLinus Torvalds 20371da177e4SLinus Torvalds sock_reset_flag(sk, SOCK_ZAPPED); 20381da177e4SLinus Torvalds 20391da177e4SLinus Torvalds sk->sk_protocol = protocol; 20401da177e4SLinus Torvalds 20411da177e4SLinus Torvalds sock->state = SS_UNCONNECTED; 20421da177e4SLinus Torvalds sk->sk_state = BT_OPEN; 20431da177e4SLinus Torvalds 20441da177e4SLinus Torvalds bt_sock_link(&hci_sk_list, sk); 20451da177e4SLinus Torvalds return 0; 20461da177e4SLinus Torvalds } 20471da177e4SLinus Torvalds 2048ec1b4cf7SStephen Hemminger static const struct net_proto_family hci_sock_family_ops = { 20491da177e4SLinus Torvalds .family = PF_BLUETOOTH, 20501da177e4SLinus Torvalds .owner = THIS_MODULE, 20511da177e4SLinus Torvalds .create = hci_sock_create, 20521da177e4SLinus Torvalds }; 20531da177e4SLinus Torvalds 20541da177e4SLinus Torvalds int __init hci_sock_init(void) 20551da177e4SLinus Torvalds { 20561da177e4SLinus Torvalds int err; 20571da177e4SLinus Torvalds 2058b0a8e282SMarcel Holtmann BUILD_BUG_ON(sizeof(struct sockaddr_hci) > sizeof(struct sockaddr)); 2059b0a8e282SMarcel Holtmann 20601da177e4SLinus Torvalds err = proto_register(&hci_sk_proto, 0); 20611da177e4SLinus Torvalds if (err < 0) 20621da177e4SLinus Torvalds return err; 20631da177e4SLinus Torvalds 20641da177e4SLinus Torvalds err = bt_sock_register(BTPROTO_HCI, &hci_sock_family_ops); 2065f7c86637SMasatake YAMATO if (err < 0) { 2066f7c86637SMasatake YAMATO BT_ERR("HCI socket registration failed"); 20671da177e4SLinus Torvalds goto error; 2068f7c86637SMasatake YAMATO } 2069f7c86637SMasatake YAMATO 2070b0316615SAl Viro err = bt_procfs_init(&init_net, "hci", &hci_sk_list, NULL); 2071f7c86637SMasatake YAMATO if (err < 0) { 2072f7c86637SMasatake YAMATO BT_ERR("Failed to create HCI proc file"); 2073f7c86637SMasatake YAMATO bt_sock_unregister(BTPROTO_HCI); 2074f7c86637SMasatake YAMATO goto error; 2075f7c86637SMasatake YAMATO } 20761da177e4SLinus Torvalds 20771da177e4SLinus Torvalds BT_INFO("HCI socket layer initialized"); 20781da177e4SLinus Torvalds 20791da177e4SLinus Torvalds return 0; 20801da177e4SLinus Torvalds 20811da177e4SLinus Torvalds error: 20821da177e4SLinus Torvalds proto_unregister(&hci_sk_proto); 20831da177e4SLinus Torvalds return err; 20841da177e4SLinus Torvalds } 20851da177e4SLinus Torvalds 2086b7440a14SAnand Gadiyar void hci_sock_cleanup(void) 20871da177e4SLinus Torvalds { 2088f7c86637SMasatake YAMATO bt_procfs_cleanup(&init_net, "hci"); 20895e9d7f86SDavid Herrmann bt_sock_unregister(BTPROTO_HCI); 20901da177e4SLinus Torvalds proto_unregister(&hci_sk_proto); 20911da177e4SLinus Torvalds } 2092