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; 5532929e1fSAlain Michaud __u8 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 && 214cc974003SMarcel Holtmann hci_skb_pkt_type(skb) != HCI_SCODATA_PKT && 215cc974003SMarcel Holtmann hci_skb_pkt_type(skb) != HCI_ISODATA_PKT) 216bb77543eSMarcel Holtmann continue; 217f81fe64fSMarcel Holtmann if (is_filtered_packet(sk, skb)) 2181da177e4SLinus Torvalds continue; 21923500189SMarcel Holtmann } else if (hci_pi(sk)->channel == HCI_CHANNEL_USER) { 22023500189SMarcel Holtmann if (!bt_cb(skb)->incoming) 22123500189SMarcel Holtmann continue; 222d79f34e3SMarcel Holtmann if (hci_skb_pkt_type(skb) != HCI_EVENT_PKT && 223d79f34e3SMarcel Holtmann hci_skb_pkt_type(skb) != HCI_ACLDATA_PKT && 224cc974003SMarcel Holtmann hci_skb_pkt_type(skb) != HCI_SCODATA_PKT && 225cc974003SMarcel Holtmann hci_skb_pkt_type(skb) != HCI_ISODATA_PKT) 22623500189SMarcel Holtmann continue; 22723500189SMarcel Holtmann } else { 22823500189SMarcel Holtmann /* Don't send frame to other channel types */ 22923500189SMarcel Holtmann continue; 23023500189SMarcel Holtmann } 2311da177e4SLinus Torvalds 232e0edf373SMarcel Holtmann if (!skb_copy) { 233e0edf373SMarcel Holtmann /* Create a private copy with headroom */ 234bad93e9dSOctavian Purdila skb_copy = __pskb_copy_fclone(skb, 1, GFP_ATOMIC, true); 235e0edf373SMarcel Holtmann if (!skb_copy) 2361da177e4SLinus Torvalds continue; 2371da177e4SLinus Torvalds 2381da177e4SLinus Torvalds /* Put type byte before the data */ 239d79f34e3SMarcel Holtmann memcpy(skb_push(skb_copy, 1), &hci_skb_pkt_type(skb), 1); 240e0edf373SMarcel Holtmann } 241e0edf373SMarcel Holtmann 242e0edf373SMarcel Holtmann nskb = skb_clone(skb_copy, GFP_ATOMIC); 243e0edf373SMarcel Holtmann if (!nskb) 244e0edf373SMarcel Holtmann continue; 2451da177e4SLinus Torvalds 2461da177e4SLinus Torvalds if (sock_queue_rcv_skb(sk, nskb)) 2471da177e4SLinus Torvalds kfree_skb(nskb); 2481da177e4SLinus Torvalds } 249470fe1b5SMarcel Holtmann 250470fe1b5SMarcel Holtmann read_unlock(&hci_sk_list.lock); 251e0edf373SMarcel Holtmann 252e0edf373SMarcel Holtmann kfree_skb(skb_copy); 253470fe1b5SMarcel Holtmann } 254470fe1b5SMarcel Holtmann 2557129069eSJohan Hedberg /* Send frame to sockets with specific channel */ 256a9ee77afSSebastian Andrzej Siewior static void __hci_send_to_channel(unsigned short channel, struct sk_buff *skb, 257c08b1a1dSMarcel Holtmann int flag, struct sock *skip_sk) 258470fe1b5SMarcel Holtmann { 259470fe1b5SMarcel Holtmann struct sock *sk; 260470fe1b5SMarcel Holtmann 2617129069eSJohan Hedberg BT_DBG("channel %u len %d", channel, skb->len); 262470fe1b5SMarcel Holtmann 263b67bfe0dSSasha Levin sk_for_each(sk, &hci_sk_list.head) { 264470fe1b5SMarcel Holtmann struct sk_buff *nskb; 265470fe1b5SMarcel Holtmann 266c08b1a1dSMarcel Holtmann /* Ignore socket without the flag set */ 267c85be545SMarcel Holtmann if (!hci_sock_test_flag(sk, flag)) 268c08b1a1dSMarcel Holtmann continue; 269c08b1a1dSMarcel Holtmann 270470fe1b5SMarcel Holtmann /* Skip the original socket */ 271470fe1b5SMarcel Holtmann if (sk == skip_sk) 272470fe1b5SMarcel Holtmann continue; 273470fe1b5SMarcel Holtmann 274470fe1b5SMarcel Holtmann if (sk->sk_state != BT_BOUND) 275470fe1b5SMarcel Holtmann continue; 276470fe1b5SMarcel Holtmann 2777129069eSJohan Hedberg if (hci_pi(sk)->channel != channel) 278d7f72f61SMarcel Holtmann continue; 279d7f72f61SMarcel Holtmann 280d7f72f61SMarcel Holtmann nskb = skb_clone(skb, GFP_ATOMIC); 281d7f72f61SMarcel Holtmann if (!nskb) 282d7f72f61SMarcel Holtmann continue; 283d7f72f61SMarcel Holtmann 284d7f72f61SMarcel Holtmann if (sock_queue_rcv_skb(sk, nskb)) 285d7f72f61SMarcel Holtmann kfree_skb(nskb); 286d7f72f61SMarcel Holtmann } 287d7f72f61SMarcel Holtmann 288a9ee77afSSebastian Andrzej Siewior } 289a9ee77afSSebastian Andrzej Siewior 290a9ee77afSSebastian Andrzej Siewior void hci_send_to_channel(unsigned short channel, struct sk_buff *skb, 291a9ee77afSSebastian Andrzej Siewior int flag, struct sock *skip_sk) 292a9ee77afSSebastian Andrzej Siewior { 293a9ee77afSSebastian Andrzej Siewior read_lock(&hci_sk_list.lock); 294a9ee77afSSebastian Andrzej Siewior __hci_send_to_channel(channel, skb, flag, skip_sk); 295d7f72f61SMarcel Holtmann read_unlock(&hci_sk_list.lock); 296d7f72f61SMarcel Holtmann } 297d7f72f61SMarcel Holtmann 298cd82e61cSMarcel Holtmann /* Send frame to monitor socket */ 299cd82e61cSMarcel Holtmann void hci_send_to_monitor(struct hci_dev *hdev, struct sk_buff *skb) 300cd82e61cSMarcel Holtmann { 301cd82e61cSMarcel Holtmann struct sk_buff *skb_copy = NULL; 3022b531294SMarcel Holtmann struct hci_mon_hdr *hdr; 303cd82e61cSMarcel Holtmann __le16 opcode; 304cd82e61cSMarcel Holtmann 305cd82e61cSMarcel Holtmann if (!atomic_read(&monitor_promisc)) 306cd82e61cSMarcel Holtmann return; 307cd82e61cSMarcel Holtmann 308cd82e61cSMarcel Holtmann BT_DBG("hdev %p len %d", hdev, skb->len); 309cd82e61cSMarcel Holtmann 310d79f34e3SMarcel Holtmann switch (hci_skb_pkt_type(skb)) { 311cd82e61cSMarcel Holtmann case HCI_COMMAND_PKT: 312dcf4adbfSJoe Perches opcode = cpu_to_le16(HCI_MON_COMMAND_PKT); 313cd82e61cSMarcel Holtmann break; 314cd82e61cSMarcel Holtmann case HCI_EVENT_PKT: 315dcf4adbfSJoe Perches opcode = cpu_to_le16(HCI_MON_EVENT_PKT); 316cd82e61cSMarcel Holtmann break; 317cd82e61cSMarcel Holtmann case HCI_ACLDATA_PKT: 318cd82e61cSMarcel Holtmann if (bt_cb(skb)->incoming) 319dcf4adbfSJoe Perches opcode = cpu_to_le16(HCI_MON_ACL_RX_PKT); 320cd82e61cSMarcel Holtmann else 321dcf4adbfSJoe Perches opcode = cpu_to_le16(HCI_MON_ACL_TX_PKT); 322cd82e61cSMarcel Holtmann break; 323cd82e61cSMarcel Holtmann case HCI_SCODATA_PKT: 324cd82e61cSMarcel Holtmann if (bt_cb(skb)->incoming) 325dcf4adbfSJoe Perches opcode = cpu_to_le16(HCI_MON_SCO_RX_PKT); 326cd82e61cSMarcel Holtmann else 327dcf4adbfSJoe Perches opcode = cpu_to_le16(HCI_MON_SCO_TX_PKT); 328cd82e61cSMarcel Holtmann break; 329f9a619dbSLuiz Augusto von Dentz case HCI_ISODATA_PKT: 330f9a619dbSLuiz Augusto von Dentz if (bt_cb(skb)->incoming) 331f9a619dbSLuiz Augusto von Dentz opcode = cpu_to_le16(HCI_MON_ISO_RX_PKT); 332f9a619dbSLuiz Augusto von Dentz else 333f9a619dbSLuiz Augusto von Dentz opcode = cpu_to_le16(HCI_MON_ISO_TX_PKT); 334f9a619dbSLuiz Augusto von Dentz break; 335e875ff84SMarcel Holtmann case HCI_DIAG_PKT: 336e875ff84SMarcel Holtmann opcode = cpu_to_le16(HCI_MON_VENDOR_DIAG); 337e875ff84SMarcel Holtmann break; 338cd82e61cSMarcel Holtmann default: 339cd82e61cSMarcel Holtmann return; 340cd82e61cSMarcel Holtmann } 341cd82e61cSMarcel Holtmann 3422b531294SMarcel Holtmann /* Create a private copy with headroom */ 3432b531294SMarcel Holtmann skb_copy = __pskb_copy_fclone(skb, HCI_MON_HDR_SIZE, GFP_ATOMIC, true); 3442b531294SMarcel Holtmann if (!skb_copy) 3452b531294SMarcel Holtmann return; 3462b531294SMarcel Holtmann 3472b531294SMarcel Holtmann /* Put header before the data */ 348d58ff351SJohannes Berg hdr = skb_push(skb_copy, HCI_MON_HDR_SIZE); 3492b531294SMarcel Holtmann hdr->opcode = opcode; 3502b531294SMarcel Holtmann hdr->index = cpu_to_le16(hdev->id); 3512b531294SMarcel Holtmann hdr->len = cpu_to_le16(skb->len); 3522b531294SMarcel Holtmann 353c08b1a1dSMarcel Holtmann hci_send_to_channel(HCI_CHANNEL_MONITOR, skb_copy, 354c08b1a1dSMarcel Holtmann HCI_SOCK_TRUSTED, NULL); 355cd82e61cSMarcel Holtmann kfree_skb(skb_copy); 356cd82e61cSMarcel Holtmann } 357cd82e61cSMarcel Holtmann 35838ceaa00SMarcel Holtmann void hci_send_monitor_ctrl_event(struct hci_dev *hdev, u16 event, 35938ceaa00SMarcel Holtmann void *data, u16 data_len, ktime_t tstamp, 36038ceaa00SMarcel Holtmann int flag, struct sock *skip_sk) 36138ceaa00SMarcel Holtmann { 36238ceaa00SMarcel Holtmann struct sock *sk; 36338ceaa00SMarcel Holtmann __le16 index; 36438ceaa00SMarcel Holtmann 36538ceaa00SMarcel Holtmann if (hdev) 36638ceaa00SMarcel Holtmann index = cpu_to_le16(hdev->id); 36738ceaa00SMarcel Holtmann else 36838ceaa00SMarcel Holtmann index = cpu_to_le16(MGMT_INDEX_NONE); 36938ceaa00SMarcel Holtmann 37038ceaa00SMarcel Holtmann read_lock(&hci_sk_list.lock); 37138ceaa00SMarcel Holtmann 37238ceaa00SMarcel Holtmann sk_for_each(sk, &hci_sk_list.head) { 37338ceaa00SMarcel Holtmann struct hci_mon_hdr *hdr; 37438ceaa00SMarcel Holtmann struct sk_buff *skb; 37538ceaa00SMarcel Holtmann 37638ceaa00SMarcel Holtmann if (hci_pi(sk)->channel != HCI_CHANNEL_CONTROL) 37738ceaa00SMarcel Holtmann continue; 37838ceaa00SMarcel Holtmann 37938ceaa00SMarcel Holtmann /* Ignore socket without the flag set */ 38038ceaa00SMarcel Holtmann if (!hci_sock_test_flag(sk, flag)) 38138ceaa00SMarcel Holtmann continue; 38238ceaa00SMarcel Holtmann 38338ceaa00SMarcel Holtmann /* Skip the original socket */ 38438ceaa00SMarcel Holtmann if (sk == skip_sk) 38538ceaa00SMarcel Holtmann continue; 38638ceaa00SMarcel Holtmann 38738ceaa00SMarcel Holtmann skb = bt_skb_alloc(6 + data_len, GFP_ATOMIC); 38838ceaa00SMarcel Holtmann if (!skb) 38938ceaa00SMarcel Holtmann continue; 39038ceaa00SMarcel Holtmann 39138ceaa00SMarcel Holtmann put_unaligned_le32(hci_pi(sk)->cookie, skb_put(skb, 4)); 39238ceaa00SMarcel Holtmann put_unaligned_le16(event, skb_put(skb, 2)); 39338ceaa00SMarcel Holtmann 39438ceaa00SMarcel Holtmann if (data) 39559ae1d12SJohannes Berg skb_put_data(skb, data, data_len); 39638ceaa00SMarcel Holtmann 39738ceaa00SMarcel Holtmann skb->tstamp = tstamp; 39838ceaa00SMarcel Holtmann 399d58ff351SJohannes Berg hdr = skb_push(skb, HCI_MON_HDR_SIZE); 40038ceaa00SMarcel Holtmann hdr->opcode = cpu_to_le16(HCI_MON_CTRL_EVENT); 40138ceaa00SMarcel Holtmann hdr->index = index; 40238ceaa00SMarcel Holtmann hdr->len = cpu_to_le16(skb->len - HCI_MON_HDR_SIZE); 40338ceaa00SMarcel Holtmann 404a9ee77afSSebastian Andrzej Siewior __hci_send_to_channel(HCI_CHANNEL_MONITOR, skb, 40538ceaa00SMarcel Holtmann HCI_SOCK_TRUSTED, NULL); 40638ceaa00SMarcel Holtmann kfree_skb(skb); 40738ceaa00SMarcel Holtmann } 40838ceaa00SMarcel Holtmann 40938ceaa00SMarcel Holtmann read_unlock(&hci_sk_list.lock); 41038ceaa00SMarcel Holtmann } 41138ceaa00SMarcel Holtmann 412cd82e61cSMarcel Holtmann static struct sk_buff *create_monitor_event(struct hci_dev *hdev, int event) 413cd82e61cSMarcel Holtmann { 414cd82e61cSMarcel Holtmann struct hci_mon_hdr *hdr; 415cd82e61cSMarcel Holtmann struct hci_mon_new_index *ni; 4166c566dd5SMarcel Holtmann struct hci_mon_index_info *ii; 417cd82e61cSMarcel Holtmann struct sk_buff *skb; 418cd82e61cSMarcel Holtmann __le16 opcode; 419cd82e61cSMarcel Holtmann 420cd82e61cSMarcel Holtmann switch (event) { 421cd82e61cSMarcel Holtmann case HCI_DEV_REG: 422cd82e61cSMarcel Holtmann skb = bt_skb_alloc(HCI_MON_NEW_INDEX_SIZE, GFP_ATOMIC); 423cd82e61cSMarcel Holtmann if (!skb) 424cd82e61cSMarcel Holtmann return NULL; 425cd82e61cSMarcel Holtmann 4264df864c1SJohannes Berg ni = skb_put(skb, HCI_MON_NEW_INDEX_SIZE); 427cd82e61cSMarcel Holtmann ni->type = hdev->dev_type; 428cd82e61cSMarcel Holtmann ni->bus = hdev->bus; 429cd82e61cSMarcel Holtmann bacpy(&ni->bdaddr, &hdev->bdaddr); 430cd82e61cSMarcel Holtmann memcpy(ni->name, hdev->name, 8); 431cd82e61cSMarcel Holtmann 432dcf4adbfSJoe Perches opcode = cpu_to_le16(HCI_MON_NEW_INDEX); 433cd82e61cSMarcel Holtmann break; 434cd82e61cSMarcel Holtmann 435cd82e61cSMarcel Holtmann case HCI_DEV_UNREG: 436cd82e61cSMarcel Holtmann skb = bt_skb_alloc(0, GFP_ATOMIC); 437cd82e61cSMarcel Holtmann if (!skb) 438cd82e61cSMarcel Holtmann return NULL; 439cd82e61cSMarcel Holtmann 440dcf4adbfSJoe Perches opcode = cpu_to_le16(HCI_MON_DEL_INDEX); 441cd82e61cSMarcel Holtmann break; 442cd82e61cSMarcel Holtmann 443e131d74aSMarcel Holtmann case HCI_DEV_SETUP: 444e131d74aSMarcel Holtmann if (hdev->manufacturer == 0xffff) 445e131d74aSMarcel Holtmann return NULL; 44619186c7bSGustavo A. R. Silva fallthrough; 447e131d74aSMarcel Holtmann 4486c566dd5SMarcel Holtmann case HCI_DEV_UP: 4496c566dd5SMarcel Holtmann skb = bt_skb_alloc(HCI_MON_INDEX_INFO_SIZE, GFP_ATOMIC); 4506c566dd5SMarcel Holtmann if (!skb) 4516c566dd5SMarcel Holtmann return NULL; 4526c566dd5SMarcel Holtmann 4534df864c1SJohannes Berg ii = skb_put(skb, HCI_MON_INDEX_INFO_SIZE); 4546c566dd5SMarcel Holtmann bacpy(&ii->bdaddr, &hdev->bdaddr); 4556c566dd5SMarcel Holtmann ii->manufacturer = cpu_to_le16(hdev->manufacturer); 4566c566dd5SMarcel Holtmann 4576c566dd5SMarcel Holtmann opcode = cpu_to_le16(HCI_MON_INDEX_INFO); 4586c566dd5SMarcel Holtmann break; 4596c566dd5SMarcel Holtmann 46022db3cbcSMarcel Holtmann case HCI_DEV_OPEN: 46122db3cbcSMarcel Holtmann skb = bt_skb_alloc(0, GFP_ATOMIC); 46222db3cbcSMarcel Holtmann if (!skb) 46322db3cbcSMarcel Holtmann return NULL; 46422db3cbcSMarcel Holtmann 46522db3cbcSMarcel Holtmann opcode = cpu_to_le16(HCI_MON_OPEN_INDEX); 46622db3cbcSMarcel Holtmann break; 46722db3cbcSMarcel Holtmann 46822db3cbcSMarcel Holtmann case HCI_DEV_CLOSE: 46922db3cbcSMarcel Holtmann skb = bt_skb_alloc(0, GFP_ATOMIC); 47022db3cbcSMarcel Holtmann if (!skb) 47122db3cbcSMarcel Holtmann return NULL; 47222db3cbcSMarcel Holtmann 47322db3cbcSMarcel Holtmann opcode = cpu_to_le16(HCI_MON_CLOSE_INDEX); 47422db3cbcSMarcel Holtmann break; 47522db3cbcSMarcel Holtmann 476cd82e61cSMarcel Holtmann default: 477cd82e61cSMarcel Holtmann return NULL; 478cd82e61cSMarcel Holtmann } 479cd82e61cSMarcel Holtmann 480cd82e61cSMarcel Holtmann __net_timestamp(skb); 481cd82e61cSMarcel Holtmann 482d58ff351SJohannes Berg hdr = skb_push(skb, HCI_MON_HDR_SIZE); 483cd82e61cSMarcel Holtmann hdr->opcode = opcode; 484cd82e61cSMarcel Holtmann hdr->index = cpu_to_le16(hdev->id); 485cd82e61cSMarcel Holtmann hdr->len = cpu_to_le16(skb->len - HCI_MON_HDR_SIZE); 486cd82e61cSMarcel Holtmann 487cd82e61cSMarcel Holtmann return skb; 488cd82e61cSMarcel Holtmann } 489cd82e61cSMarcel Holtmann 490249fa169SMarcel Holtmann static struct sk_buff *create_monitor_ctrl_open(struct sock *sk) 491249fa169SMarcel Holtmann { 492249fa169SMarcel Holtmann struct hci_mon_hdr *hdr; 493249fa169SMarcel Holtmann struct sk_buff *skb; 494d0bef1d2SMarcel Holtmann u16 format; 495249fa169SMarcel Holtmann u8 ver[3]; 496249fa169SMarcel Holtmann u32 flags; 497249fa169SMarcel Holtmann 4980ef2c42fSMarcel Holtmann /* No message needed when cookie is not present */ 4990ef2c42fSMarcel Holtmann if (!hci_pi(sk)->cookie) 5000ef2c42fSMarcel Holtmann return NULL; 5010ef2c42fSMarcel Holtmann 502d0bef1d2SMarcel Holtmann switch (hci_pi(sk)->channel) { 503f81f5b2dSMarcel Holtmann case HCI_CHANNEL_RAW: 504f81f5b2dSMarcel Holtmann format = 0x0000; 505f81f5b2dSMarcel Holtmann ver[0] = BT_SUBSYS_VERSION; 506f81f5b2dSMarcel Holtmann put_unaligned_le16(BT_SUBSYS_REVISION, ver + 1); 507f81f5b2dSMarcel Holtmann break; 508aa1638ddSMarcel Holtmann case HCI_CHANNEL_USER: 509aa1638ddSMarcel Holtmann format = 0x0001; 510aa1638ddSMarcel Holtmann ver[0] = BT_SUBSYS_VERSION; 511aa1638ddSMarcel Holtmann put_unaligned_le16(BT_SUBSYS_REVISION, ver + 1); 512aa1638ddSMarcel Holtmann break; 513d0bef1d2SMarcel Holtmann case HCI_CHANNEL_CONTROL: 514d0bef1d2SMarcel Holtmann format = 0x0002; 515d0bef1d2SMarcel Holtmann mgmt_fill_version_info(ver); 516d0bef1d2SMarcel Holtmann break; 517d0bef1d2SMarcel Holtmann default: 518d0bef1d2SMarcel Holtmann /* No message for unsupported format */ 519d0bef1d2SMarcel Holtmann return NULL; 520d0bef1d2SMarcel Holtmann } 521d0bef1d2SMarcel Holtmann 522249fa169SMarcel Holtmann skb = bt_skb_alloc(14 + TASK_COMM_LEN , GFP_ATOMIC); 523249fa169SMarcel Holtmann if (!skb) 524249fa169SMarcel Holtmann return NULL; 525249fa169SMarcel Holtmann 526249fa169SMarcel Holtmann flags = hci_sock_test_flag(sk, HCI_SOCK_TRUSTED) ? 0x1 : 0x0; 527249fa169SMarcel Holtmann 528249fa169SMarcel Holtmann put_unaligned_le32(hci_pi(sk)->cookie, skb_put(skb, 4)); 529249fa169SMarcel Holtmann put_unaligned_le16(format, skb_put(skb, 2)); 53059ae1d12SJohannes Berg skb_put_data(skb, ver, sizeof(ver)); 531249fa169SMarcel Holtmann put_unaligned_le32(flags, skb_put(skb, 4)); 532634fef61SJohannes Berg skb_put_u8(skb, TASK_COMM_LEN); 53359ae1d12SJohannes Berg skb_put_data(skb, hci_pi(sk)->comm, TASK_COMM_LEN); 534249fa169SMarcel Holtmann 535249fa169SMarcel Holtmann __net_timestamp(skb); 536249fa169SMarcel Holtmann 537d58ff351SJohannes Berg hdr = skb_push(skb, HCI_MON_HDR_SIZE); 538249fa169SMarcel Holtmann hdr->opcode = cpu_to_le16(HCI_MON_CTRL_OPEN); 5390ef2c42fSMarcel Holtmann if (hci_pi(sk)->hdev) 5400ef2c42fSMarcel Holtmann hdr->index = cpu_to_le16(hci_pi(sk)->hdev->id); 5410ef2c42fSMarcel Holtmann else 542249fa169SMarcel Holtmann hdr->index = cpu_to_le16(HCI_DEV_NONE); 543249fa169SMarcel Holtmann hdr->len = cpu_to_le16(skb->len - HCI_MON_HDR_SIZE); 544249fa169SMarcel Holtmann 545249fa169SMarcel Holtmann return skb; 546249fa169SMarcel Holtmann } 547249fa169SMarcel Holtmann 548249fa169SMarcel Holtmann static struct sk_buff *create_monitor_ctrl_close(struct sock *sk) 549249fa169SMarcel Holtmann { 550249fa169SMarcel Holtmann struct hci_mon_hdr *hdr; 551249fa169SMarcel Holtmann struct sk_buff *skb; 552249fa169SMarcel Holtmann 5530ef2c42fSMarcel Holtmann /* No message needed when cookie is not present */ 5540ef2c42fSMarcel Holtmann if (!hci_pi(sk)->cookie) 5550ef2c42fSMarcel Holtmann return NULL; 5560ef2c42fSMarcel Holtmann 557d0bef1d2SMarcel Holtmann switch (hci_pi(sk)->channel) { 558f81f5b2dSMarcel Holtmann case HCI_CHANNEL_RAW: 559aa1638ddSMarcel Holtmann case HCI_CHANNEL_USER: 560d0bef1d2SMarcel Holtmann case HCI_CHANNEL_CONTROL: 561d0bef1d2SMarcel Holtmann break; 562d0bef1d2SMarcel Holtmann default: 563d0bef1d2SMarcel Holtmann /* No message for unsupported format */ 564d0bef1d2SMarcel Holtmann return NULL; 565d0bef1d2SMarcel Holtmann } 566d0bef1d2SMarcel Holtmann 567249fa169SMarcel Holtmann skb = bt_skb_alloc(4, GFP_ATOMIC); 568249fa169SMarcel Holtmann if (!skb) 569249fa169SMarcel Holtmann return NULL; 570249fa169SMarcel Holtmann 571249fa169SMarcel Holtmann put_unaligned_le32(hci_pi(sk)->cookie, skb_put(skb, 4)); 572249fa169SMarcel Holtmann 573249fa169SMarcel Holtmann __net_timestamp(skb); 574249fa169SMarcel Holtmann 575d58ff351SJohannes Berg hdr = skb_push(skb, HCI_MON_HDR_SIZE); 576249fa169SMarcel Holtmann hdr->opcode = cpu_to_le16(HCI_MON_CTRL_CLOSE); 5770ef2c42fSMarcel Holtmann if (hci_pi(sk)->hdev) 5780ef2c42fSMarcel Holtmann hdr->index = cpu_to_le16(hci_pi(sk)->hdev->id); 5790ef2c42fSMarcel Holtmann else 580249fa169SMarcel Holtmann hdr->index = cpu_to_le16(HCI_DEV_NONE); 581249fa169SMarcel Holtmann hdr->len = cpu_to_le16(skb->len - HCI_MON_HDR_SIZE); 582249fa169SMarcel Holtmann 583249fa169SMarcel Holtmann return skb; 584249fa169SMarcel Holtmann } 585249fa169SMarcel Holtmann 58638ceaa00SMarcel Holtmann static struct sk_buff *create_monitor_ctrl_command(struct sock *sk, u16 index, 58738ceaa00SMarcel Holtmann u16 opcode, u16 len, 58838ceaa00SMarcel Holtmann const void *buf) 58938ceaa00SMarcel Holtmann { 59038ceaa00SMarcel Holtmann struct hci_mon_hdr *hdr; 59138ceaa00SMarcel Holtmann struct sk_buff *skb; 59238ceaa00SMarcel Holtmann 59338ceaa00SMarcel Holtmann skb = bt_skb_alloc(6 + len, GFP_ATOMIC); 59438ceaa00SMarcel Holtmann if (!skb) 59538ceaa00SMarcel Holtmann return NULL; 59638ceaa00SMarcel Holtmann 59738ceaa00SMarcel Holtmann put_unaligned_le32(hci_pi(sk)->cookie, skb_put(skb, 4)); 59838ceaa00SMarcel Holtmann put_unaligned_le16(opcode, skb_put(skb, 2)); 59938ceaa00SMarcel Holtmann 60038ceaa00SMarcel Holtmann if (buf) 60159ae1d12SJohannes Berg skb_put_data(skb, buf, len); 60238ceaa00SMarcel Holtmann 60338ceaa00SMarcel Holtmann __net_timestamp(skb); 60438ceaa00SMarcel Holtmann 605d58ff351SJohannes Berg hdr = skb_push(skb, HCI_MON_HDR_SIZE); 60638ceaa00SMarcel Holtmann hdr->opcode = cpu_to_le16(HCI_MON_CTRL_COMMAND); 60738ceaa00SMarcel Holtmann hdr->index = cpu_to_le16(index); 60838ceaa00SMarcel Holtmann hdr->len = cpu_to_le16(skb->len - HCI_MON_HDR_SIZE); 60938ceaa00SMarcel Holtmann 61038ceaa00SMarcel Holtmann return skb; 61138ceaa00SMarcel Holtmann } 61238ceaa00SMarcel Holtmann 613787b306cSJohannes Berg static void __printf(2, 3) 614787b306cSJohannes Berg send_monitor_note(struct sock *sk, const char *fmt, ...) 615dd31506dSMarcel Holtmann { 616787b306cSJohannes Berg size_t len; 617dd31506dSMarcel Holtmann struct hci_mon_hdr *hdr; 618dd31506dSMarcel Holtmann struct sk_buff *skb; 619787b306cSJohannes Berg va_list args; 620787b306cSJohannes Berg 621787b306cSJohannes Berg va_start(args, fmt); 622787b306cSJohannes Berg len = vsnprintf(NULL, 0, fmt, args); 623787b306cSJohannes Berg va_end(args); 624dd31506dSMarcel Holtmann 625dd31506dSMarcel Holtmann skb = bt_skb_alloc(len + 1, GFP_ATOMIC); 626dd31506dSMarcel Holtmann if (!skb) 627dd31506dSMarcel Holtmann return; 628dd31506dSMarcel Holtmann 629787b306cSJohannes Berg va_start(args, fmt); 630787b306cSJohannes Berg vsprintf(skb_put(skb, len), fmt, args); 6314df864c1SJohannes Berg *(u8 *)skb_put(skb, 1) = 0; 632787b306cSJohannes Berg va_end(args); 633dd31506dSMarcel Holtmann 634dd31506dSMarcel Holtmann __net_timestamp(skb); 635dd31506dSMarcel Holtmann 636dd31506dSMarcel Holtmann hdr = (void *)skb_push(skb, HCI_MON_HDR_SIZE); 637dd31506dSMarcel Holtmann hdr->opcode = cpu_to_le16(HCI_MON_SYSTEM_NOTE); 638dd31506dSMarcel Holtmann hdr->index = cpu_to_le16(HCI_DEV_NONE); 639dd31506dSMarcel Holtmann hdr->len = cpu_to_le16(skb->len - HCI_MON_HDR_SIZE); 640dd31506dSMarcel Holtmann 641dd31506dSMarcel Holtmann if (sock_queue_rcv_skb(sk, skb)) 642dd31506dSMarcel Holtmann kfree_skb(skb); 643dd31506dSMarcel Holtmann } 644dd31506dSMarcel Holtmann 645cd82e61cSMarcel Holtmann static void send_monitor_replay(struct sock *sk) 646cd82e61cSMarcel Holtmann { 647cd82e61cSMarcel Holtmann struct hci_dev *hdev; 648cd82e61cSMarcel Holtmann 649cd82e61cSMarcel Holtmann read_lock(&hci_dev_list_lock); 650cd82e61cSMarcel Holtmann 651cd82e61cSMarcel Holtmann list_for_each_entry(hdev, &hci_dev_list, list) { 652cd82e61cSMarcel Holtmann struct sk_buff *skb; 653cd82e61cSMarcel Holtmann 654cd82e61cSMarcel Holtmann skb = create_monitor_event(hdev, HCI_DEV_REG); 655cd82e61cSMarcel Holtmann if (!skb) 656cd82e61cSMarcel Holtmann continue; 657cd82e61cSMarcel Holtmann 658cd82e61cSMarcel Holtmann if (sock_queue_rcv_skb(sk, skb)) 659cd82e61cSMarcel Holtmann kfree_skb(skb); 66022db3cbcSMarcel Holtmann 66122db3cbcSMarcel Holtmann if (!test_bit(HCI_RUNNING, &hdev->flags)) 66222db3cbcSMarcel Holtmann continue; 66322db3cbcSMarcel Holtmann 66422db3cbcSMarcel Holtmann skb = create_monitor_event(hdev, HCI_DEV_OPEN); 66522db3cbcSMarcel Holtmann if (!skb) 66622db3cbcSMarcel Holtmann continue; 66722db3cbcSMarcel Holtmann 66822db3cbcSMarcel Holtmann if (sock_queue_rcv_skb(sk, skb)) 66922db3cbcSMarcel Holtmann kfree_skb(skb); 6706c566dd5SMarcel Holtmann 671e131d74aSMarcel Holtmann if (test_bit(HCI_UP, &hdev->flags)) 6726c566dd5SMarcel Holtmann skb = create_monitor_event(hdev, HCI_DEV_UP); 673e131d74aSMarcel Holtmann else if (hci_dev_test_flag(hdev, HCI_SETUP)) 674e131d74aSMarcel Holtmann skb = create_monitor_event(hdev, HCI_DEV_SETUP); 675e131d74aSMarcel Holtmann else 676e131d74aSMarcel Holtmann skb = NULL; 6776c566dd5SMarcel Holtmann 678e131d74aSMarcel Holtmann if (skb) { 6796c566dd5SMarcel Holtmann if (sock_queue_rcv_skb(sk, skb)) 6806c566dd5SMarcel Holtmann kfree_skb(skb); 681cd82e61cSMarcel Holtmann } 682e131d74aSMarcel Holtmann } 683cd82e61cSMarcel Holtmann 684cd82e61cSMarcel Holtmann read_unlock(&hci_dev_list_lock); 685cd82e61cSMarcel Holtmann } 686cd82e61cSMarcel Holtmann 687249fa169SMarcel Holtmann static void send_monitor_control_replay(struct sock *mon_sk) 688249fa169SMarcel Holtmann { 689249fa169SMarcel Holtmann struct sock *sk; 690249fa169SMarcel Holtmann 691249fa169SMarcel Holtmann read_lock(&hci_sk_list.lock); 692249fa169SMarcel Holtmann 693249fa169SMarcel Holtmann sk_for_each(sk, &hci_sk_list.head) { 694249fa169SMarcel Holtmann struct sk_buff *skb; 695249fa169SMarcel Holtmann 696249fa169SMarcel Holtmann skb = create_monitor_ctrl_open(sk); 697249fa169SMarcel Holtmann if (!skb) 698249fa169SMarcel Holtmann continue; 699249fa169SMarcel Holtmann 700249fa169SMarcel Holtmann if (sock_queue_rcv_skb(mon_sk, skb)) 701249fa169SMarcel Holtmann kfree_skb(skb); 702249fa169SMarcel Holtmann } 703249fa169SMarcel Holtmann 704249fa169SMarcel Holtmann read_unlock(&hci_sk_list.lock); 705249fa169SMarcel Holtmann } 706249fa169SMarcel Holtmann 707040030efSMarcel Holtmann /* Generate internal stack event */ 708040030efSMarcel Holtmann static void hci_si_event(struct hci_dev *hdev, int type, int dlen, void *data) 709040030efSMarcel Holtmann { 710040030efSMarcel Holtmann struct hci_event_hdr *hdr; 711040030efSMarcel Holtmann struct hci_ev_stack_internal *ev; 712040030efSMarcel Holtmann struct sk_buff *skb; 713040030efSMarcel Holtmann 714040030efSMarcel Holtmann skb = bt_skb_alloc(HCI_EVENT_HDR_SIZE + sizeof(*ev) + dlen, GFP_ATOMIC); 715040030efSMarcel Holtmann if (!skb) 716040030efSMarcel Holtmann return; 717040030efSMarcel Holtmann 7184df864c1SJohannes Berg hdr = skb_put(skb, HCI_EVENT_HDR_SIZE); 719040030efSMarcel Holtmann hdr->evt = HCI_EV_STACK_INTERNAL; 720040030efSMarcel Holtmann hdr->plen = sizeof(*ev) + dlen; 721040030efSMarcel Holtmann 7224df864c1SJohannes Berg ev = skb_put(skb, sizeof(*ev) + dlen); 723040030efSMarcel Holtmann ev->type = type; 724040030efSMarcel Holtmann memcpy(ev->data, data, dlen); 725040030efSMarcel Holtmann 726040030efSMarcel Holtmann bt_cb(skb)->incoming = 1; 727040030efSMarcel Holtmann __net_timestamp(skb); 728040030efSMarcel Holtmann 729d79f34e3SMarcel Holtmann hci_skb_pkt_type(skb) = HCI_EVENT_PKT; 730040030efSMarcel Holtmann hci_send_to_sock(hdev, skb); 731040030efSMarcel Holtmann kfree_skb(skb); 732040030efSMarcel Holtmann } 733040030efSMarcel Holtmann 734040030efSMarcel Holtmann void hci_sock_dev_event(struct hci_dev *hdev, int event) 735040030efSMarcel Holtmann { 736040030efSMarcel Holtmann BT_DBG("hdev %s event %d", hdev->name, event); 737040030efSMarcel Holtmann 738cd82e61cSMarcel Holtmann if (atomic_read(&monitor_promisc)) { 739cd82e61cSMarcel Holtmann struct sk_buff *skb; 740cd82e61cSMarcel Holtmann 741ed1b28a4SMarcel Holtmann /* Send event to monitor */ 742cd82e61cSMarcel Holtmann skb = create_monitor_event(hdev, event); 743cd82e61cSMarcel Holtmann if (skb) { 744c08b1a1dSMarcel Holtmann hci_send_to_channel(HCI_CHANNEL_MONITOR, skb, 745c08b1a1dSMarcel Holtmann HCI_SOCK_TRUSTED, NULL); 746cd82e61cSMarcel Holtmann kfree_skb(skb); 747cd82e61cSMarcel Holtmann } 748cd82e61cSMarcel Holtmann } 749cd82e61cSMarcel Holtmann 750ed1b28a4SMarcel Holtmann if (event <= HCI_DEV_DOWN) { 751ed1b28a4SMarcel Holtmann struct hci_ev_si_device ev; 752ed1b28a4SMarcel Holtmann 753040030efSMarcel Holtmann /* Send event to sockets */ 754040030efSMarcel Holtmann ev.event = event; 755040030efSMarcel Holtmann ev.dev_id = hdev->id; 756040030efSMarcel Holtmann hci_si_event(NULL, HCI_EV_SI_DEVICE, sizeof(ev), &ev); 757ed1b28a4SMarcel Holtmann } 758040030efSMarcel Holtmann 759040030efSMarcel Holtmann if (event == HCI_DEV_UNREG) { 760040030efSMarcel Holtmann struct sock *sk; 761040030efSMarcel Holtmann 762040030efSMarcel Holtmann /* Detach sockets from device */ 763040030efSMarcel Holtmann read_lock(&hci_sk_list.lock); 764b67bfe0dSSasha Levin sk_for_each(sk, &hci_sk_list.head) { 765040030efSMarcel Holtmann bh_lock_sock_nested(sk); 766040030efSMarcel Holtmann if (hci_pi(sk)->hdev == hdev) { 767040030efSMarcel Holtmann hci_pi(sk)->hdev = NULL; 768040030efSMarcel Holtmann sk->sk_err = EPIPE; 769040030efSMarcel Holtmann sk->sk_state = BT_OPEN; 770040030efSMarcel Holtmann sk->sk_state_change(sk); 771040030efSMarcel Holtmann 772040030efSMarcel Holtmann hci_dev_put(hdev); 773040030efSMarcel Holtmann } 774040030efSMarcel Holtmann bh_unlock_sock(sk); 775040030efSMarcel Holtmann } 776040030efSMarcel Holtmann read_unlock(&hci_sk_list.lock); 777040030efSMarcel Holtmann } 778040030efSMarcel Holtmann } 779040030efSMarcel Holtmann 780801c1e8dSJohan Hedberg static struct hci_mgmt_chan *__hci_mgmt_chan_find(unsigned short channel) 781801c1e8dSJohan Hedberg { 782801c1e8dSJohan Hedberg struct hci_mgmt_chan *c; 783801c1e8dSJohan Hedberg 784801c1e8dSJohan Hedberg list_for_each_entry(c, &mgmt_chan_list, list) { 785801c1e8dSJohan Hedberg if (c->channel == channel) 786801c1e8dSJohan Hedberg return c; 787801c1e8dSJohan Hedberg } 788801c1e8dSJohan Hedberg 789801c1e8dSJohan Hedberg return NULL; 790801c1e8dSJohan Hedberg } 791801c1e8dSJohan Hedberg 792801c1e8dSJohan Hedberg static struct hci_mgmt_chan *hci_mgmt_chan_find(unsigned short channel) 793801c1e8dSJohan Hedberg { 794801c1e8dSJohan Hedberg struct hci_mgmt_chan *c; 795801c1e8dSJohan Hedberg 796801c1e8dSJohan Hedberg mutex_lock(&mgmt_chan_list_lock); 797801c1e8dSJohan Hedberg c = __hci_mgmt_chan_find(channel); 798801c1e8dSJohan Hedberg mutex_unlock(&mgmt_chan_list_lock); 799801c1e8dSJohan Hedberg 800801c1e8dSJohan Hedberg return c; 801801c1e8dSJohan Hedberg } 802801c1e8dSJohan Hedberg 803801c1e8dSJohan Hedberg int hci_mgmt_chan_register(struct hci_mgmt_chan *c) 804801c1e8dSJohan Hedberg { 805801c1e8dSJohan Hedberg if (c->channel < HCI_CHANNEL_CONTROL) 806801c1e8dSJohan Hedberg return -EINVAL; 807801c1e8dSJohan Hedberg 808801c1e8dSJohan Hedberg mutex_lock(&mgmt_chan_list_lock); 809801c1e8dSJohan Hedberg if (__hci_mgmt_chan_find(c->channel)) { 810801c1e8dSJohan Hedberg mutex_unlock(&mgmt_chan_list_lock); 811801c1e8dSJohan Hedberg return -EALREADY; 812801c1e8dSJohan Hedberg } 813801c1e8dSJohan Hedberg 814801c1e8dSJohan Hedberg list_add_tail(&c->list, &mgmt_chan_list); 815801c1e8dSJohan Hedberg 816801c1e8dSJohan Hedberg mutex_unlock(&mgmt_chan_list_lock); 817801c1e8dSJohan Hedberg 818801c1e8dSJohan Hedberg return 0; 819801c1e8dSJohan Hedberg } 820801c1e8dSJohan Hedberg EXPORT_SYMBOL(hci_mgmt_chan_register); 821801c1e8dSJohan Hedberg 822801c1e8dSJohan Hedberg void hci_mgmt_chan_unregister(struct hci_mgmt_chan *c) 823801c1e8dSJohan Hedberg { 824801c1e8dSJohan Hedberg mutex_lock(&mgmt_chan_list_lock); 825801c1e8dSJohan Hedberg list_del(&c->list); 826801c1e8dSJohan Hedberg mutex_unlock(&mgmt_chan_list_lock); 827801c1e8dSJohan Hedberg } 828801c1e8dSJohan Hedberg EXPORT_SYMBOL(hci_mgmt_chan_unregister); 829801c1e8dSJohan Hedberg 8301da177e4SLinus Torvalds static int hci_sock_release(struct socket *sock) 8311da177e4SLinus Torvalds { 8321da177e4SLinus Torvalds struct sock *sk = sock->sk; 8337b005bd3SMarcel Holtmann struct hci_dev *hdev; 834249fa169SMarcel Holtmann struct sk_buff *skb; 8351da177e4SLinus Torvalds 8361da177e4SLinus Torvalds BT_DBG("sock %p sk %p", sock, sk); 8371da177e4SLinus Torvalds 8381da177e4SLinus Torvalds if (!sk) 8391da177e4SLinus Torvalds return 0; 8401da177e4SLinus Torvalds 84111eb85ecSDan Carpenter lock_sock(sk); 84211eb85ecSDan Carpenter 84370ecce91SMarcel Holtmann switch (hci_pi(sk)->channel) { 84470ecce91SMarcel Holtmann case HCI_CHANNEL_MONITOR: 845cd82e61cSMarcel Holtmann atomic_dec(&monitor_promisc); 84670ecce91SMarcel Holtmann break; 847f81f5b2dSMarcel Holtmann case HCI_CHANNEL_RAW: 848aa1638ddSMarcel Holtmann case HCI_CHANNEL_USER: 84970ecce91SMarcel Holtmann case HCI_CHANNEL_CONTROL: 850249fa169SMarcel Holtmann /* Send event to monitor */ 851249fa169SMarcel Holtmann skb = create_monitor_ctrl_close(sk); 852249fa169SMarcel Holtmann if (skb) { 853249fa169SMarcel Holtmann hci_send_to_channel(HCI_CHANNEL_MONITOR, skb, 854249fa169SMarcel Holtmann HCI_SOCK_TRUSTED, NULL); 855249fa169SMarcel Holtmann kfree_skb(skb); 856249fa169SMarcel Holtmann } 857249fa169SMarcel Holtmann 858df1cb87aSMarcel Holtmann hci_sock_free_cookie(sk); 85970ecce91SMarcel Holtmann break; 86070ecce91SMarcel Holtmann } 861cd82e61cSMarcel Holtmann 8621da177e4SLinus Torvalds bt_sock_unlink(&hci_sk_list, sk); 8631da177e4SLinus Torvalds 864e20a2e9cSMyungho Jung hdev = hci_pi(sk)->hdev; 8651da177e4SLinus Torvalds if (hdev) { 86623500189SMarcel Holtmann if (hci_pi(sk)->channel == HCI_CHANNEL_USER) { 8679332ef9dSMasahiro Yamada /* When releasing a user channel exclusive access, 8686b3cc1dbSSimon Fels * call hci_dev_do_close directly instead of calling 8696b3cc1dbSSimon Fels * hci_dev_close to ensure the exclusive access will 8706b3cc1dbSSimon Fels * be released and the controller brought back down. 8716b3cc1dbSSimon Fels * 8726b3cc1dbSSimon Fels * The checking of HCI_AUTO_OFF is not needed in this 8736b3cc1dbSSimon Fels * case since it will have been cleared already when 8746b3cc1dbSSimon Fels * opening the user channel. 8756b3cc1dbSSimon Fels */ 8766b3cc1dbSSimon Fels hci_dev_do_close(hdev); 8779380f9eaSLoic Poulain hci_dev_clear_flag(hdev, HCI_USER_CHANNEL); 8789380f9eaSLoic Poulain mgmt_index_added(hdev); 87923500189SMarcel Holtmann } 88023500189SMarcel Holtmann 8811da177e4SLinus Torvalds atomic_dec(&hdev->promisc); 8821da177e4SLinus Torvalds hci_dev_put(hdev); 8831da177e4SLinus Torvalds } 8841da177e4SLinus Torvalds 8851da177e4SLinus Torvalds sock_orphan(sk); 8861da177e4SLinus Torvalds 8871da177e4SLinus Torvalds skb_queue_purge(&sk->sk_receive_queue); 8881da177e4SLinus Torvalds skb_queue_purge(&sk->sk_write_queue); 8891da177e4SLinus Torvalds 89011eb85ecSDan Carpenter release_sock(sk); 8911da177e4SLinus Torvalds sock_put(sk); 8921da177e4SLinus Torvalds return 0; 8931da177e4SLinus Torvalds } 8941da177e4SLinus Torvalds 895b2a66aadSAntti Julku static int hci_sock_blacklist_add(struct hci_dev *hdev, void __user *arg) 896f0358568SJohan Hedberg { 897f0358568SJohan Hedberg bdaddr_t bdaddr; 8985e762444SAntti Julku int err; 899f0358568SJohan Hedberg 900f0358568SJohan Hedberg if (copy_from_user(&bdaddr, arg, sizeof(bdaddr))) 901f0358568SJohan Hedberg return -EFAULT; 902f0358568SJohan Hedberg 90309fd0de5SGustavo F. Padovan hci_dev_lock(hdev); 9045e762444SAntti Julku 905dcc36c16SJohan Hedberg err = hci_bdaddr_list_add(&hdev->blacklist, &bdaddr, BDADDR_BREDR); 9065e762444SAntti Julku 90709fd0de5SGustavo F. Padovan hci_dev_unlock(hdev); 9085e762444SAntti Julku 9095e762444SAntti Julku return err; 910f0358568SJohan Hedberg } 911f0358568SJohan Hedberg 912b2a66aadSAntti Julku static int hci_sock_blacklist_del(struct hci_dev *hdev, void __user *arg) 913f0358568SJohan Hedberg { 914f0358568SJohan Hedberg bdaddr_t bdaddr; 9155e762444SAntti Julku int err; 916f0358568SJohan Hedberg 917f0358568SJohan Hedberg if (copy_from_user(&bdaddr, arg, sizeof(bdaddr))) 918f0358568SJohan Hedberg return -EFAULT; 919f0358568SJohan Hedberg 92009fd0de5SGustavo F. Padovan hci_dev_lock(hdev); 9215e762444SAntti Julku 922dcc36c16SJohan Hedberg err = hci_bdaddr_list_del(&hdev->blacklist, &bdaddr, BDADDR_BREDR); 9235e762444SAntti Julku 92409fd0de5SGustavo F. Padovan hci_dev_unlock(hdev); 9255e762444SAntti Julku 9265e762444SAntti Julku return err; 927f0358568SJohan Hedberg } 928f0358568SJohan Hedberg 9291da177e4SLinus Torvalds /* Ioctls that require bound socket */ 9306039aa73SGustavo Padovan static int hci_sock_bound_ioctl(struct sock *sk, unsigned int cmd, 9316039aa73SGustavo Padovan unsigned long arg) 9321da177e4SLinus Torvalds { 9331da177e4SLinus Torvalds struct hci_dev *hdev = hci_pi(sk)->hdev; 9341da177e4SLinus Torvalds 9351da177e4SLinus Torvalds if (!hdev) 9361da177e4SLinus Torvalds return -EBADFD; 9371da177e4SLinus Torvalds 938d7a5a11dSMarcel Holtmann if (hci_dev_test_flag(hdev, HCI_USER_CHANNEL)) 9390736cfa8SMarcel Holtmann return -EBUSY; 9400736cfa8SMarcel Holtmann 941d7a5a11dSMarcel Holtmann if (hci_dev_test_flag(hdev, HCI_UNCONFIGURED)) 942fee746b0SMarcel Holtmann return -EOPNOTSUPP; 943fee746b0SMarcel Holtmann 944ca8bee5dSMarcel Holtmann if (hdev->dev_type != HCI_PRIMARY) 9455b69bef5SMarcel Holtmann return -EOPNOTSUPP; 9465b69bef5SMarcel Holtmann 9471da177e4SLinus Torvalds switch (cmd) { 9481da177e4SLinus Torvalds case HCISETRAW: 9491da177e4SLinus Torvalds if (!capable(CAP_NET_ADMIN)) 950bf5b30b8SZhao Hongjiang return -EPERM; 951db596681SMarcel Holtmann return -EOPNOTSUPP; 9521da177e4SLinus Torvalds 9531da177e4SLinus Torvalds case HCIGETCONNINFO: 9541da177e4SLinus Torvalds return hci_get_conn_info(hdev, (void __user *)arg); 9551da177e4SLinus Torvalds 95640be492fSMarcel Holtmann case HCIGETAUTHINFO: 95740be492fSMarcel Holtmann return hci_get_auth_info(hdev, (void __user *)arg); 95840be492fSMarcel Holtmann 959f0358568SJohan Hedberg case HCIBLOCKADDR: 960f0358568SJohan Hedberg if (!capable(CAP_NET_ADMIN)) 961bf5b30b8SZhao Hongjiang return -EPERM; 962b2a66aadSAntti Julku return hci_sock_blacklist_add(hdev, (void __user *)arg); 963f0358568SJohan Hedberg 964f0358568SJohan Hedberg case HCIUNBLOCKADDR: 965f0358568SJohan Hedberg if (!capable(CAP_NET_ADMIN)) 966bf5b30b8SZhao Hongjiang return -EPERM; 967b2a66aadSAntti Julku return hci_sock_blacklist_del(hdev, (void __user *)arg); 9680736cfa8SMarcel Holtmann } 969f0358568SJohan Hedberg 970324d36edSMarcel Holtmann return -ENOIOCTLCMD; 9711da177e4SLinus Torvalds } 9721da177e4SLinus Torvalds 9738fc9ced3SGustavo Padovan static int hci_sock_ioctl(struct socket *sock, unsigned int cmd, 9748fc9ced3SGustavo Padovan unsigned long arg) 9751da177e4SLinus Torvalds { 9761da177e4SLinus Torvalds void __user *argp = (void __user *)arg; 9770736cfa8SMarcel Holtmann struct sock *sk = sock->sk; 9781da177e4SLinus Torvalds int err; 9791da177e4SLinus Torvalds 9801da177e4SLinus Torvalds BT_DBG("cmd %x arg %lx", cmd, arg); 9811da177e4SLinus Torvalds 982c1c4f956SMarcel Holtmann lock_sock(sk); 983c1c4f956SMarcel Holtmann 984c1c4f956SMarcel Holtmann if (hci_pi(sk)->channel != HCI_CHANNEL_RAW) { 985c1c4f956SMarcel Holtmann err = -EBADFD; 986c1c4f956SMarcel Holtmann goto done; 987c1c4f956SMarcel Holtmann } 988c1c4f956SMarcel Holtmann 989f81f5b2dSMarcel Holtmann /* When calling an ioctl on an unbound raw socket, then ensure 990f81f5b2dSMarcel Holtmann * that the monitor gets informed. Ensure that the resulting event 991f81f5b2dSMarcel Holtmann * is only send once by checking if the cookie exists or not. The 992f81f5b2dSMarcel Holtmann * socket cookie will be only ever generated once for the lifetime 993f81f5b2dSMarcel Holtmann * of a given socket. 994f81f5b2dSMarcel Holtmann */ 995f81f5b2dSMarcel Holtmann if (hci_sock_gen_cookie(sk)) { 996f81f5b2dSMarcel Holtmann struct sk_buff *skb; 997f81f5b2dSMarcel Holtmann 998f81f5b2dSMarcel Holtmann if (capable(CAP_NET_ADMIN)) 999f81f5b2dSMarcel Holtmann hci_sock_set_flag(sk, HCI_SOCK_TRUSTED); 1000f81f5b2dSMarcel Holtmann 1001f81f5b2dSMarcel Holtmann /* Send event to monitor */ 1002f81f5b2dSMarcel Holtmann skb = create_monitor_ctrl_open(sk); 1003f81f5b2dSMarcel Holtmann if (skb) { 1004f81f5b2dSMarcel Holtmann hci_send_to_channel(HCI_CHANNEL_MONITOR, skb, 1005f81f5b2dSMarcel Holtmann HCI_SOCK_TRUSTED, NULL); 1006f81f5b2dSMarcel Holtmann kfree_skb(skb); 1007f81f5b2dSMarcel Holtmann } 1008f81f5b2dSMarcel Holtmann } 1009f81f5b2dSMarcel Holtmann 1010c1c4f956SMarcel Holtmann release_sock(sk); 1011c1c4f956SMarcel Holtmann 10121da177e4SLinus Torvalds switch (cmd) { 10131da177e4SLinus Torvalds case HCIGETDEVLIST: 10141da177e4SLinus Torvalds return hci_get_dev_list(argp); 10151da177e4SLinus Torvalds 10161da177e4SLinus Torvalds case HCIGETDEVINFO: 10171da177e4SLinus Torvalds return hci_get_dev_info(argp); 10181da177e4SLinus Torvalds 10191da177e4SLinus Torvalds case HCIGETCONNLIST: 10201da177e4SLinus Torvalds return hci_get_conn_list(argp); 10211da177e4SLinus Torvalds 10221da177e4SLinus Torvalds case HCIDEVUP: 10231da177e4SLinus Torvalds if (!capable(CAP_NET_ADMIN)) 1024bf5b30b8SZhao Hongjiang return -EPERM; 10251da177e4SLinus Torvalds return hci_dev_open(arg); 10261da177e4SLinus Torvalds 10271da177e4SLinus Torvalds case HCIDEVDOWN: 10281da177e4SLinus Torvalds if (!capable(CAP_NET_ADMIN)) 1029bf5b30b8SZhao Hongjiang return -EPERM; 10301da177e4SLinus Torvalds return hci_dev_close(arg); 10311da177e4SLinus Torvalds 10321da177e4SLinus Torvalds case HCIDEVRESET: 10331da177e4SLinus Torvalds if (!capable(CAP_NET_ADMIN)) 1034bf5b30b8SZhao Hongjiang return -EPERM; 10351da177e4SLinus Torvalds return hci_dev_reset(arg); 10361da177e4SLinus Torvalds 10371da177e4SLinus Torvalds case HCIDEVRESTAT: 10381da177e4SLinus Torvalds if (!capable(CAP_NET_ADMIN)) 1039bf5b30b8SZhao Hongjiang return -EPERM; 10401da177e4SLinus Torvalds return hci_dev_reset_stat(arg); 10411da177e4SLinus Torvalds 10421da177e4SLinus Torvalds case HCISETSCAN: 10431da177e4SLinus Torvalds case HCISETAUTH: 10441da177e4SLinus Torvalds case HCISETENCRYPT: 10451da177e4SLinus Torvalds case HCISETPTYPE: 10461da177e4SLinus Torvalds case HCISETLINKPOL: 10471da177e4SLinus Torvalds case HCISETLINKMODE: 10481da177e4SLinus Torvalds case HCISETACLMTU: 10491da177e4SLinus Torvalds case HCISETSCOMTU: 10501da177e4SLinus Torvalds if (!capable(CAP_NET_ADMIN)) 1051bf5b30b8SZhao Hongjiang return -EPERM; 10521da177e4SLinus Torvalds return hci_dev_cmd(cmd, argp); 10531da177e4SLinus Torvalds 10541da177e4SLinus Torvalds case HCIINQUIRY: 10551da177e4SLinus Torvalds return hci_inquiry(argp); 1056c1c4f956SMarcel Holtmann } 10571da177e4SLinus Torvalds 10581da177e4SLinus Torvalds lock_sock(sk); 1059c1c4f956SMarcel Holtmann 10601da177e4SLinus Torvalds err = hci_sock_bound_ioctl(sk, cmd, arg); 1061c1c4f956SMarcel Holtmann 1062c1c4f956SMarcel Holtmann done: 10631da177e4SLinus Torvalds release_sock(sk); 10641da177e4SLinus Torvalds return err; 10651da177e4SLinus Torvalds } 10661da177e4SLinus Torvalds 10677a6038b3SArnd Bergmann #ifdef CONFIG_COMPAT 10687a6038b3SArnd Bergmann static int hci_sock_compat_ioctl(struct socket *sock, unsigned int cmd, 10697a6038b3SArnd Bergmann unsigned long arg) 10707a6038b3SArnd Bergmann { 10717a6038b3SArnd Bergmann switch (cmd) { 10727a6038b3SArnd Bergmann case HCIDEVUP: 10737a6038b3SArnd Bergmann case HCIDEVDOWN: 10747a6038b3SArnd Bergmann case HCIDEVRESET: 10757a6038b3SArnd Bergmann case HCIDEVRESTAT: 10767a6038b3SArnd Bergmann return hci_sock_ioctl(sock, cmd, arg); 10777a6038b3SArnd Bergmann } 10787a6038b3SArnd Bergmann 10797a6038b3SArnd Bergmann return hci_sock_ioctl(sock, cmd, (unsigned long)compat_ptr(arg)); 10807a6038b3SArnd Bergmann } 10817a6038b3SArnd Bergmann #endif 10827a6038b3SArnd Bergmann 10838fc9ced3SGustavo Padovan static int hci_sock_bind(struct socket *sock, struct sockaddr *addr, 10848fc9ced3SGustavo Padovan int addr_len) 10851da177e4SLinus Torvalds { 10860381101fSJohan Hedberg struct sockaddr_hci haddr; 10871da177e4SLinus Torvalds struct sock *sk = sock->sk; 10881da177e4SLinus Torvalds struct hci_dev *hdev = NULL; 1089f4cdbb3fSMarcel Holtmann struct sk_buff *skb; 10900381101fSJohan Hedberg int len, err = 0; 10911da177e4SLinus Torvalds 10921da177e4SLinus Torvalds BT_DBG("sock %p sk %p", sock, sk); 10931da177e4SLinus Torvalds 10940381101fSJohan Hedberg if (!addr) 10950381101fSJohan Hedberg return -EINVAL; 10960381101fSJohan Hedberg 10970381101fSJohan Hedberg memset(&haddr, 0, sizeof(haddr)); 10980381101fSJohan Hedberg len = min_t(unsigned int, sizeof(haddr), addr_len); 10990381101fSJohan Hedberg memcpy(&haddr, addr, len); 11000381101fSJohan Hedberg 11010381101fSJohan Hedberg if (haddr.hci_family != AF_BLUETOOTH) 11020381101fSJohan Hedberg return -EINVAL; 11030381101fSJohan Hedberg 11041da177e4SLinus Torvalds lock_sock(sk); 11051da177e4SLinus Torvalds 11067cc2ade2SMarcel Holtmann if (sk->sk_state == BT_BOUND) { 11077cc2ade2SMarcel Holtmann err = -EALREADY; 11087cc2ade2SMarcel Holtmann goto done; 11097cc2ade2SMarcel Holtmann } 11107cc2ade2SMarcel Holtmann 11117cc2ade2SMarcel Holtmann switch (haddr.hci_channel) { 11127cc2ade2SMarcel Holtmann case HCI_CHANNEL_RAW: 11137cc2ade2SMarcel Holtmann if (hci_pi(sk)->hdev) { 11141da177e4SLinus Torvalds err = -EALREADY; 11151da177e4SLinus Torvalds goto done; 11161da177e4SLinus Torvalds } 11171da177e4SLinus Torvalds 11180381101fSJohan Hedberg if (haddr.hci_dev != HCI_DEV_NONE) { 11190381101fSJohan Hedberg hdev = hci_dev_get(haddr.hci_dev); 112070f23020SAndrei Emeltchenko if (!hdev) { 11211da177e4SLinus Torvalds err = -ENODEV; 11221da177e4SLinus Torvalds goto done; 11231da177e4SLinus Torvalds } 11241da177e4SLinus Torvalds 11251da177e4SLinus Torvalds atomic_inc(&hdev->promisc); 11261da177e4SLinus Torvalds } 11271da177e4SLinus Torvalds 11285a6d2cf5SMarcel Holtmann hci_pi(sk)->channel = haddr.hci_channel; 1129f81f5b2dSMarcel Holtmann 1130f4cdbb3fSMarcel Holtmann if (!hci_sock_gen_cookie(sk)) { 1131f4cdbb3fSMarcel Holtmann /* In the case when a cookie has already been assigned, 1132f4cdbb3fSMarcel Holtmann * then there has been already an ioctl issued against 1133*91641b79SZheng Yongjun * an unbound socket and with that triggered an open 1134f4cdbb3fSMarcel Holtmann * notification. Send a close notification first to 1135f4cdbb3fSMarcel Holtmann * allow the state transition to bounded. 1136f81f5b2dSMarcel Holtmann */ 1137f4cdbb3fSMarcel Holtmann skb = create_monitor_ctrl_close(sk); 1138f4cdbb3fSMarcel Holtmann if (skb) { 1139f4cdbb3fSMarcel Holtmann hci_send_to_channel(HCI_CHANNEL_MONITOR, skb, 1140f4cdbb3fSMarcel Holtmann HCI_SOCK_TRUSTED, NULL); 1141f4cdbb3fSMarcel Holtmann kfree_skb(skb); 1142f4cdbb3fSMarcel Holtmann } 1143f4cdbb3fSMarcel Holtmann } 1144f81f5b2dSMarcel Holtmann 1145f81f5b2dSMarcel Holtmann if (capable(CAP_NET_ADMIN)) 1146f81f5b2dSMarcel Holtmann hci_sock_set_flag(sk, HCI_SOCK_TRUSTED); 1147f81f5b2dSMarcel Holtmann 1148f4cdbb3fSMarcel Holtmann hci_pi(sk)->hdev = hdev; 1149f4cdbb3fSMarcel Holtmann 1150f81f5b2dSMarcel Holtmann /* Send event to monitor */ 1151f81f5b2dSMarcel Holtmann skb = create_monitor_ctrl_open(sk); 1152f81f5b2dSMarcel Holtmann if (skb) { 1153f81f5b2dSMarcel Holtmann hci_send_to_channel(HCI_CHANNEL_MONITOR, skb, 1154f81f5b2dSMarcel Holtmann HCI_SOCK_TRUSTED, NULL); 1155f81f5b2dSMarcel Holtmann kfree_skb(skb); 1156f81f5b2dSMarcel Holtmann } 11577cc2ade2SMarcel Holtmann break; 11587cc2ade2SMarcel Holtmann 115923500189SMarcel Holtmann case HCI_CHANNEL_USER: 116023500189SMarcel Holtmann if (hci_pi(sk)->hdev) { 116123500189SMarcel Holtmann err = -EALREADY; 116223500189SMarcel Holtmann goto done; 116323500189SMarcel Holtmann } 116423500189SMarcel Holtmann 116523500189SMarcel Holtmann if (haddr.hci_dev == HCI_DEV_NONE) { 116623500189SMarcel Holtmann err = -EINVAL; 116723500189SMarcel Holtmann goto done; 116823500189SMarcel Holtmann } 116923500189SMarcel Holtmann 117010a8b86fSMarcel Holtmann if (!capable(CAP_NET_ADMIN)) { 117123500189SMarcel Holtmann err = -EPERM; 117223500189SMarcel Holtmann goto done; 117323500189SMarcel Holtmann } 117423500189SMarcel Holtmann 117523500189SMarcel Holtmann hdev = hci_dev_get(haddr.hci_dev); 117623500189SMarcel Holtmann if (!hdev) { 117723500189SMarcel Holtmann err = -ENODEV; 117823500189SMarcel Holtmann goto done; 117923500189SMarcel Holtmann } 118023500189SMarcel Holtmann 1181781f899fSMarcel Holtmann if (test_bit(HCI_INIT, &hdev->flags) || 1182d7a5a11dSMarcel Holtmann hci_dev_test_flag(hdev, HCI_SETUP) || 1183781f899fSMarcel Holtmann hci_dev_test_flag(hdev, HCI_CONFIG) || 1184781f899fSMarcel Holtmann (!hci_dev_test_flag(hdev, HCI_AUTO_OFF) && 1185781f899fSMarcel Holtmann test_bit(HCI_UP, &hdev->flags))) { 118623500189SMarcel Holtmann err = -EBUSY; 118723500189SMarcel Holtmann hci_dev_put(hdev); 118823500189SMarcel Holtmann goto done; 118923500189SMarcel Holtmann } 119023500189SMarcel Holtmann 1191238be788SMarcel Holtmann if (hci_dev_test_and_set_flag(hdev, HCI_USER_CHANNEL)) { 119223500189SMarcel Holtmann err = -EUSERS; 119323500189SMarcel Holtmann hci_dev_put(hdev); 119423500189SMarcel Holtmann goto done; 119523500189SMarcel Holtmann } 119623500189SMarcel Holtmann 119723500189SMarcel Holtmann mgmt_index_removed(hdev); 119823500189SMarcel Holtmann 119923500189SMarcel Holtmann err = hci_dev_open(hdev->id); 120023500189SMarcel Holtmann if (err) { 1201781f899fSMarcel Holtmann if (err == -EALREADY) { 1202781f899fSMarcel Holtmann /* In case the transport is already up and 1203781f899fSMarcel Holtmann * running, clear the error here. 1204781f899fSMarcel Holtmann * 12059332ef9dSMasahiro Yamada * This can happen when opening a user 1206781f899fSMarcel Holtmann * channel and HCI_AUTO_OFF grace period 1207781f899fSMarcel Holtmann * is still active. 1208781f899fSMarcel Holtmann */ 1209781f899fSMarcel Holtmann err = 0; 1210781f899fSMarcel Holtmann } else { 1211a358dc11SMarcel Holtmann hci_dev_clear_flag(hdev, HCI_USER_CHANNEL); 1212c6521401SMarcel Holtmann mgmt_index_added(hdev); 121323500189SMarcel Holtmann hci_dev_put(hdev); 121423500189SMarcel Holtmann goto done; 121523500189SMarcel Holtmann } 1216781f899fSMarcel Holtmann } 121723500189SMarcel Holtmann 12185a6d2cf5SMarcel Holtmann hci_pi(sk)->channel = haddr.hci_channel; 1219aa1638ddSMarcel Holtmann 1220aa1638ddSMarcel Holtmann if (!hci_sock_gen_cookie(sk)) { 1221aa1638ddSMarcel Holtmann /* In the case when a cookie has already been assigned, 1222aa1638ddSMarcel Holtmann * this socket will transition from a raw socket into 12239332ef9dSMasahiro Yamada * a user channel socket. For a clean transition, send 1224aa1638ddSMarcel Holtmann * the close notification first. 1225aa1638ddSMarcel Holtmann */ 1226aa1638ddSMarcel Holtmann skb = create_monitor_ctrl_close(sk); 1227aa1638ddSMarcel Holtmann if (skb) { 1228aa1638ddSMarcel Holtmann hci_send_to_channel(HCI_CHANNEL_MONITOR, skb, 1229aa1638ddSMarcel Holtmann HCI_SOCK_TRUSTED, NULL); 1230aa1638ddSMarcel Holtmann kfree_skb(skb); 1231aa1638ddSMarcel Holtmann } 1232aa1638ddSMarcel Holtmann } 1233aa1638ddSMarcel Holtmann 1234aa1638ddSMarcel Holtmann /* The user channel is restricted to CAP_NET_ADMIN 1235aa1638ddSMarcel Holtmann * capabilities and with that implicitly trusted. 1236aa1638ddSMarcel Holtmann */ 1237aa1638ddSMarcel Holtmann hci_sock_set_flag(sk, HCI_SOCK_TRUSTED); 1238aa1638ddSMarcel Holtmann 123923500189SMarcel Holtmann hci_pi(sk)->hdev = hdev; 12405a6d2cf5SMarcel Holtmann 1241aa1638ddSMarcel Holtmann /* Send event to monitor */ 1242aa1638ddSMarcel Holtmann skb = create_monitor_ctrl_open(sk); 1243aa1638ddSMarcel Holtmann if (skb) { 1244aa1638ddSMarcel Holtmann hci_send_to_channel(HCI_CHANNEL_MONITOR, skb, 1245aa1638ddSMarcel Holtmann HCI_SOCK_TRUSTED, NULL); 1246aa1638ddSMarcel Holtmann kfree_skb(skb); 1247aa1638ddSMarcel Holtmann } 1248aa1638ddSMarcel Holtmann 12495a6d2cf5SMarcel Holtmann atomic_inc(&hdev->promisc); 125023500189SMarcel Holtmann break; 125123500189SMarcel Holtmann 1252cd82e61cSMarcel Holtmann case HCI_CHANNEL_MONITOR: 1253cd82e61cSMarcel Holtmann if (haddr.hci_dev != HCI_DEV_NONE) { 1254cd82e61cSMarcel Holtmann err = -EINVAL; 1255cd82e61cSMarcel Holtmann goto done; 1256cd82e61cSMarcel Holtmann } 1257cd82e61cSMarcel Holtmann 1258cd82e61cSMarcel Holtmann if (!capable(CAP_NET_RAW)) { 1259cd82e61cSMarcel Holtmann err = -EPERM; 1260cd82e61cSMarcel Holtmann goto done; 1261cd82e61cSMarcel Holtmann } 1262cd82e61cSMarcel Holtmann 12635a6d2cf5SMarcel Holtmann hci_pi(sk)->channel = haddr.hci_channel; 12645a6d2cf5SMarcel Holtmann 126550ebc055SMarcel Holtmann /* The monitor interface is restricted to CAP_NET_RAW 126650ebc055SMarcel Holtmann * capabilities and with that implicitly trusted. 126750ebc055SMarcel Holtmann */ 126850ebc055SMarcel Holtmann hci_sock_set_flag(sk, HCI_SOCK_TRUSTED); 126950ebc055SMarcel Holtmann 1270787b306cSJohannes Berg send_monitor_note(sk, "Linux version %s (%s)", 1271787b306cSJohannes Berg init_utsname()->release, 1272787b306cSJohannes Berg init_utsname()->machine); 12739e8305b3SMarcel Holtmann send_monitor_note(sk, "Bluetooth subsystem version %u.%u", 12749e8305b3SMarcel Holtmann BT_SUBSYS_VERSION, BT_SUBSYS_REVISION); 1275cd82e61cSMarcel Holtmann send_monitor_replay(sk); 1276249fa169SMarcel Holtmann send_monitor_control_replay(sk); 1277cd82e61cSMarcel Holtmann 1278cd82e61cSMarcel Holtmann atomic_inc(&monitor_promisc); 1279cd82e61cSMarcel Holtmann break; 1280cd82e61cSMarcel Holtmann 1281ac714949SMarcel Holtmann case HCI_CHANNEL_LOGGING: 1282ac714949SMarcel Holtmann if (haddr.hci_dev != HCI_DEV_NONE) { 1283ac714949SMarcel Holtmann err = -EINVAL; 1284ac714949SMarcel Holtmann goto done; 1285ac714949SMarcel Holtmann } 1286ac714949SMarcel Holtmann 1287ac714949SMarcel Holtmann if (!capable(CAP_NET_ADMIN)) { 1288ac714949SMarcel Holtmann err = -EPERM; 1289ac714949SMarcel Holtmann goto done; 1290ac714949SMarcel Holtmann } 12915a6d2cf5SMarcel Holtmann 12925a6d2cf5SMarcel Holtmann hci_pi(sk)->channel = haddr.hci_channel; 1293ac714949SMarcel Holtmann break; 1294ac714949SMarcel Holtmann 12957cc2ade2SMarcel Holtmann default: 1296801c1e8dSJohan Hedberg if (!hci_mgmt_chan_find(haddr.hci_channel)) { 12977cc2ade2SMarcel Holtmann err = -EINVAL; 12987cc2ade2SMarcel Holtmann goto done; 12997cc2ade2SMarcel Holtmann } 13007cc2ade2SMarcel Holtmann 1301801c1e8dSJohan Hedberg if (haddr.hci_dev != HCI_DEV_NONE) { 1302801c1e8dSJohan Hedberg err = -EINVAL; 1303801c1e8dSJohan Hedberg goto done; 1304801c1e8dSJohan Hedberg } 1305801c1e8dSJohan Hedberg 13061195fbb8SMarcel Holtmann /* Users with CAP_NET_ADMIN capabilities are allowed 13071195fbb8SMarcel Holtmann * access to all management commands and events. For 13081195fbb8SMarcel Holtmann * untrusted users the interface is restricted and 13091195fbb8SMarcel Holtmann * also only untrusted events are sent. 131050ebc055SMarcel Holtmann */ 13111195fbb8SMarcel Holtmann if (capable(CAP_NET_ADMIN)) 131250ebc055SMarcel Holtmann hci_sock_set_flag(sk, HCI_SOCK_TRUSTED); 131350ebc055SMarcel Holtmann 13145a6d2cf5SMarcel Holtmann hci_pi(sk)->channel = haddr.hci_channel; 13155a6d2cf5SMarcel Holtmann 1316f9207338SMarcel Holtmann /* At the moment the index and unconfigured index events 1317f9207338SMarcel Holtmann * are enabled unconditionally. Setting them on each 1318f9207338SMarcel Holtmann * socket when binding keeps this functionality. They 1319f9207338SMarcel Holtmann * however might be cleared later and then sending of these 1320f9207338SMarcel Holtmann * events will be disabled, but that is then intentional. 1321f6b7712eSMarcel Holtmann * 1322f6b7712eSMarcel Holtmann * This also enables generic events that are safe to be 1323f6b7712eSMarcel Holtmann * received by untrusted users. Example for such events 1324f6b7712eSMarcel Holtmann * are changes to settings, class of device, name etc. 1325f9207338SMarcel Holtmann */ 13265a6d2cf5SMarcel Holtmann if (hci_pi(sk)->channel == HCI_CHANNEL_CONTROL) { 1327f4cdbb3fSMarcel Holtmann if (!hci_sock_gen_cookie(sk)) { 1328f4cdbb3fSMarcel Holtmann /* In the case when a cookie has already been 1329*91641b79SZheng Yongjun * assigned, this socket will transition from 1330f4cdbb3fSMarcel Holtmann * a raw socket into a control socket. To 1331*91641b79SZheng Yongjun * allow for a clean transition, send the 1332f4cdbb3fSMarcel Holtmann * close notification first. 1333f4cdbb3fSMarcel Holtmann */ 1334f4cdbb3fSMarcel Holtmann skb = create_monitor_ctrl_close(sk); 1335f4cdbb3fSMarcel Holtmann if (skb) { 1336f4cdbb3fSMarcel Holtmann hci_send_to_channel(HCI_CHANNEL_MONITOR, skb, 1337f4cdbb3fSMarcel Holtmann HCI_SOCK_TRUSTED, NULL); 1338f4cdbb3fSMarcel Holtmann kfree_skb(skb); 1339f4cdbb3fSMarcel Holtmann } 1340f4cdbb3fSMarcel Holtmann } 134170ecce91SMarcel Holtmann 1342249fa169SMarcel Holtmann /* Send event to monitor */ 1343249fa169SMarcel Holtmann skb = create_monitor_ctrl_open(sk); 1344249fa169SMarcel Holtmann if (skb) { 1345249fa169SMarcel Holtmann hci_send_to_channel(HCI_CHANNEL_MONITOR, skb, 1346249fa169SMarcel Holtmann HCI_SOCK_TRUSTED, NULL); 1347249fa169SMarcel Holtmann kfree_skb(skb); 1348249fa169SMarcel Holtmann } 1349249fa169SMarcel Holtmann 1350f9207338SMarcel Holtmann hci_sock_set_flag(sk, HCI_MGMT_INDEX_EVENTS); 1351f9207338SMarcel Holtmann hci_sock_set_flag(sk, HCI_MGMT_UNCONF_INDEX_EVENTS); 13525504c3a3SMarcel Holtmann hci_sock_set_flag(sk, HCI_MGMT_OPTION_EVENTS); 13535504c3a3SMarcel Holtmann hci_sock_set_flag(sk, HCI_MGMT_SETTING_EVENTS); 13545504c3a3SMarcel Holtmann hci_sock_set_flag(sk, HCI_MGMT_DEV_CLASS_EVENTS); 13555504c3a3SMarcel Holtmann hci_sock_set_flag(sk, HCI_MGMT_LOCAL_NAME_EVENTS); 1356f9207338SMarcel Holtmann } 1357801c1e8dSJohan Hedberg break; 1358801c1e8dSJohan Hedberg } 1359801c1e8dSJohan Hedberg 13601da177e4SLinus Torvalds sk->sk_state = BT_BOUND; 13611da177e4SLinus Torvalds 13621da177e4SLinus Torvalds done: 13631da177e4SLinus Torvalds release_sock(sk); 13641da177e4SLinus Torvalds return err; 13651da177e4SLinus Torvalds } 13661da177e4SLinus Torvalds 13678fc9ced3SGustavo Padovan static int hci_sock_getname(struct socket *sock, struct sockaddr *addr, 13689b2c45d4SDenys Vlasenko int peer) 13691da177e4SLinus Torvalds { 13701da177e4SLinus Torvalds struct sockaddr_hci *haddr = (struct sockaddr_hci *)addr; 13711da177e4SLinus Torvalds struct sock *sk = sock->sk; 13729d4b68b2SMarcel Holtmann struct hci_dev *hdev; 13739d4b68b2SMarcel Holtmann int err = 0; 13741da177e4SLinus Torvalds 13751da177e4SLinus Torvalds BT_DBG("sock %p sk %p", sock, sk); 13761da177e4SLinus Torvalds 137706f43cbcSMarcel Holtmann if (peer) 137806f43cbcSMarcel Holtmann return -EOPNOTSUPP; 137906f43cbcSMarcel Holtmann 13801da177e4SLinus Torvalds lock_sock(sk); 13811da177e4SLinus Torvalds 13829d4b68b2SMarcel Holtmann hdev = hci_pi(sk)->hdev; 13839d4b68b2SMarcel Holtmann if (!hdev) { 13849d4b68b2SMarcel Holtmann err = -EBADFD; 13859d4b68b2SMarcel Holtmann goto done; 13869d4b68b2SMarcel Holtmann } 13879d4b68b2SMarcel Holtmann 13881da177e4SLinus Torvalds haddr->hci_family = AF_BLUETOOTH; 13897b005bd3SMarcel Holtmann haddr->hci_dev = hdev->id; 13909d4b68b2SMarcel Holtmann haddr->hci_channel= hci_pi(sk)->channel; 13919b2c45d4SDenys Vlasenko err = sizeof(*haddr); 13921da177e4SLinus Torvalds 13939d4b68b2SMarcel Holtmann done: 13941da177e4SLinus Torvalds release_sock(sk); 13959d4b68b2SMarcel Holtmann return err; 13961da177e4SLinus Torvalds } 13971da177e4SLinus Torvalds 13986039aa73SGustavo Padovan static void hci_sock_cmsg(struct sock *sk, struct msghdr *msg, 13996039aa73SGustavo Padovan struct sk_buff *skb) 14001da177e4SLinus Torvalds { 140132929e1fSAlain Michaud __u8 mask = hci_pi(sk)->cmsg_mask; 14021da177e4SLinus Torvalds 14030d48d939SMarcel Holtmann if (mask & HCI_CMSG_DIR) { 14040d48d939SMarcel Holtmann int incoming = bt_cb(skb)->incoming; 14058fc9ced3SGustavo Padovan put_cmsg(msg, SOL_HCI, HCI_CMSG_DIR, sizeof(incoming), 14068fc9ced3SGustavo Padovan &incoming); 14070d48d939SMarcel Holtmann } 14081da177e4SLinus Torvalds 1409a61bbcf2SPatrick McHardy if (mask & HCI_CMSG_TSTAMP) { 1410f6e623a6SJohann Felix Soden #ifdef CONFIG_COMPAT 141113c6ee2aSDeepa Dinamani struct old_timeval32 ctv; 1412f6e623a6SJohann Felix Soden #endif 141313c6ee2aSDeepa Dinamani struct __kernel_old_timeval tv; 1414767c5eb5SMarcel Holtmann void *data; 1415767c5eb5SMarcel Holtmann int len; 1416a61bbcf2SPatrick McHardy 1417a61bbcf2SPatrick McHardy skb_get_timestamp(skb, &tv); 1418767c5eb5SMarcel Holtmann 14191da97f83SDavid S. Miller data = &tv; 14201da97f83SDavid S. Miller len = sizeof(tv); 14211da97f83SDavid S. Miller #ifdef CONFIG_COMPAT 1422da88cea1SH. J. Lu if (!COMPAT_USE_64BIT_TIME && 1423da88cea1SH. J. Lu (msg->msg_flags & MSG_CMSG_COMPAT)) { 1424767c5eb5SMarcel Holtmann ctv.tv_sec = tv.tv_sec; 1425767c5eb5SMarcel Holtmann ctv.tv_usec = tv.tv_usec; 1426767c5eb5SMarcel Holtmann data = &ctv; 1427767c5eb5SMarcel Holtmann len = sizeof(ctv); 1428767c5eb5SMarcel Holtmann } 14291da97f83SDavid S. Miller #endif 1430767c5eb5SMarcel Holtmann 1431767c5eb5SMarcel Holtmann put_cmsg(msg, SOL_HCI, HCI_CMSG_TSTAMP, len, data); 1432a61bbcf2SPatrick McHardy } 14331da177e4SLinus Torvalds } 14341da177e4SLinus Torvalds 14358528d3f7SMarcel Holtmann static int hci_sock_recvmsg(struct socket *sock, struct msghdr *msg, 14368528d3f7SMarcel Holtmann size_t len, int flags) 14371da177e4SLinus Torvalds { 14381da177e4SLinus Torvalds int noblock = flags & MSG_DONTWAIT; 14391da177e4SLinus Torvalds struct sock *sk = sock->sk; 14401da177e4SLinus Torvalds struct sk_buff *skb; 14411da177e4SLinus Torvalds int copied, err; 144283871f8cSDenis Kenzior unsigned int skblen; 14431da177e4SLinus Torvalds 14441da177e4SLinus Torvalds BT_DBG("sock %p, sk %p", sock, sk); 14451da177e4SLinus Torvalds 1446d94a6104SMarcel Holtmann if (flags & MSG_OOB) 14471da177e4SLinus Torvalds return -EOPNOTSUPP; 14481da177e4SLinus Torvalds 1449ac714949SMarcel Holtmann if (hci_pi(sk)->channel == HCI_CHANNEL_LOGGING) 1450ac714949SMarcel Holtmann return -EOPNOTSUPP; 1451ac714949SMarcel Holtmann 14521da177e4SLinus Torvalds if (sk->sk_state == BT_CLOSED) 14531da177e4SLinus Torvalds return 0; 14541da177e4SLinus Torvalds 145570f23020SAndrei Emeltchenko skb = skb_recv_datagram(sk, flags, noblock, &err); 145670f23020SAndrei Emeltchenko if (!skb) 14571da177e4SLinus Torvalds return err; 14581da177e4SLinus Torvalds 145983871f8cSDenis Kenzior skblen = skb->len; 14601da177e4SLinus Torvalds copied = skb->len; 14611da177e4SLinus Torvalds if (len < copied) { 14621da177e4SLinus Torvalds msg->msg_flags |= MSG_TRUNC; 14631da177e4SLinus Torvalds copied = len; 14641da177e4SLinus Torvalds } 14651da177e4SLinus Torvalds 1466badff6d0SArnaldo Carvalho de Melo skb_reset_transport_header(skb); 146751f3d02bSDavid S. Miller err = skb_copy_datagram_msg(skb, 0, msg, copied); 14681da177e4SLinus Torvalds 14693a208627SMarcel Holtmann switch (hci_pi(sk)->channel) { 14703a208627SMarcel Holtmann case HCI_CHANNEL_RAW: 14711da177e4SLinus Torvalds hci_sock_cmsg(sk, msg, skb); 14723a208627SMarcel Holtmann break; 147323500189SMarcel Holtmann case HCI_CHANNEL_USER: 1474cd82e61cSMarcel Holtmann case HCI_CHANNEL_MONITOR: 1475cd82e61cSMarcel Holtmann sock_recv_timestamp(msg, sk, skb); 1476cd82e61cSMarcel Holtmann break; 1477801c1e8dSJohan Hedberg default: 1478801c1e8dSJohan Hedberg if (hci_mgmt_chan_find(hci_pi(sk)->channel)) 1479801c1e8dSJohan Hedberg sock_recv_timestamp(msg, sk, skb); 1480801c1e8dSJohan Hedberg break; 14813a208627SMarcel Holtmann } 14821da177e4SLinus Torvalds 14831da177e4SLinus Torvalds skb_free_datagram(sk, skb); 14841da177e4SLinus Torvalds 14854f34228bSLuiz Augusto von Dentz if (flags & MSG_TRUNC) 148683871f8cSDenis Kenzior copied = skblen; 148783871f8cSDenis Kenzior 14881da177e4SLinus Torvalds return err ? : copied; 14891da177e4SLinus Torvalds } 14901da177e4SLinus Torvalds 1491fa4335d7SJohan Hedberg static int hci_mgmt_cmd(struct hci_mgmt_chan *chan, struct sock *sk, 1492fa4335d7SJohan Hedberg struct msghdr *msg, size_t msglen) 1493fa4335d7SJohan Hedberg { 1494fa4335d7SJohan Hedberg void *buf; 1495fa4335d7SJohan Hedberg u8 *cp; 1496fa4335d7SJohan Hedberg struct mgmt_hdr *hdr; 1497fa4335d7SJohan Hedberg u16 opcode, index, len; 1498fa4335d7SJohan Hedberg struct hci_dev *hdev = NULL; 1499fa4335d7SJohan Hedberg const struct hci_mgmt_handler *handler; 1500fa4335d7SJohan Hedberg bool var_len, no_hdev; 1501fa4335d7SJohan Hedberg int err; 1502fa4335d7SJohan Hedberg 1503fa4335d7SJohan Hedberg BT_DBG("got %zu bytes", msglen); 1504fa4335d7SJohan Hedberg 1505fa4335d7SJohan Hedberg if (msglen < sizeof(*hdr)) 1506fa4335d7SJohan Hedberg return -EINVAL; 1507fa4335d7SJohan Hedberg 1508fa4335d7SJohan Hedberg buf = kmalloc(msglen, GFP_KERNEL); 1509fa4335d7SJohan Hedberg if (!buf) 1510fa4335d7SJohan Hedberg return -ENOMEM; 1511fa4335d7SJohan Hedberg 1512fa4335d7SJohan Hedberg if (memcpy_from_msg(buf, msg, msglen)) { 1513fa4335d7SJohan Hedberg err = -EFAULT; 1514fa4335d7SJohan Hedberg goto done; 1515fa4335d7SJohan Hedberg } 1516fa4335d7SJohan Hedberg 1517fa4335d7SJohan Hedberg hdr = buf; 1518fa4335d7SJohan Hedberg opcode = __le16_to_cpu(hdr->opcode); 1519fa4335d7SJohan Hedberg index = __le16_to_cpu(hdr->index); 1520fa4335d7SJohan Hedberg len = __le16_to_cpu(hdr->len); 1521fa4335d7SJohan Hedberg 1522fa4335d7SJohan Hedberg if (len != msglen - sizeof(*hdr)) { 1523fa4335d7SJohan Hedberg err = -EINVAL; 1524fa4335d7SJohan Hedberg goto done; 1525fa4335d7SJohan Hedberg } 1526fa4335d7SJohan Hedberg 152738ceaa00SMarcel Holtmann if (chan->channel == HCI_CHANNEL_CONTROL) { 152838ceaa00SMarcel Holtmann struct sk_buff *skb; 152938ceaa00SMarcel Holtmann 153038ceaa00SMarcel Holtmann /* Send event to monitor */ 153138ceaa00SMarcel Holtmann skb = create_monitor_ctrl_command(sk, index, opcode, len, 153238ceaa00SMarcel Holtmann buf + sizeof(*hdr)); 153338ceaa00SMarcel Holtmann if (skb) { 153438ceaa00SMarcel Holtmann hci_send_to_channel(HCI_CHANNEL_MONITOR, skb, 153538ceaa00SMarcel Holtmann HCI_SOCK_TRUSTED, NULL); 153638ceaa00SMarcel Holtmann kfree_skb(skb); 153738ceaa00SMarcel Holtmann } 153838ceaa00SMarcel Holtmann } 153938ceaa00SMarcel Holtmann 1540fa4335d7SJohan Hedberg if (opcode >= chan->handler_count || 1541fa4335d7SJohan Hedberg chan->handlers[opcode].func == NULL) { 1542fa4335d7SJohan Hedberg BT_DBG("Unknown op %u", opcode); 1543fa4335d7SJohan Hedberg err = mgmt_cmd_status(sk, index, opcode, 1544fa4335d7SJohan Hedberg MGMT_STATUS_UNKNOWN_COMMAND); 1545fa4335d7SJohan Hedberg goto done; 1546fa4335d7SJohan Hedberg } 1547fa4335d7SJohan Hedberg 1548fa4335d7SJohan Hedberg handler = &chan->handlers[opcode]; 1549fa4335d7SJohan Hedberg 1550fa4335d7SJohan Hedberg if (!hci_sock_test_flag(sk, HCI_SOCK_TRUSTED) && 1551fa4335d7SJohan Hedberg !(handler->flags & HCI_MGMT_UNTRUSTED)) { 1552fa4335d7SJohan Hedberg err = mgmt_cmd_status(sk, index, opcode, 1553fa4335d7SJohan Hedberg MGMT_STATUS_PERMISSION_DENIED); 1554fa4335d7SJohan Hedberg goto done; 1555fa4335d7SJohan Hedberg } 1556fa4335d7SJohan Hedberg 1557fa4335d7SJohan Hedberg if (index != MGMT_INDEX_NONE) { 1558fa4335d7SJohan Hedberg hdev = hci_dev_get(index); 1559fa4335d7SJohan Hedberg if (!hdev) { 1560fa4335d7SJohan Hedberg err = mgmt_cmd_status(sk, index, opcode, 1561fa4335d7SJohan Hedberg MGMT_STATUS_INVALID_INDEX); 1562fa4335d7SJohan Hedberg goto done; 1563fa4335d7SJohan Hedberg } 1564fa4335d7SJohan Hedberg 1565fa4335d7SJohan Hedberg if (hci_dev_test_flag(hdev, HCI_SETUP) || 1566fa4335d7SJohan Hedberg hci_dev_test_flag(hdev, HCI_CONFIG) || 1567fa4335d7SJohan Hedberg hci_dev_test_flag(hdev, HCI_USER_CHANNEL)) { 1568fa4335d7SJohan Hedberg err = mgmt_cmd_status(sk, index, opcode, 1569fa4335d7SJohan Hedberg MGMT_STATUS_INVALID_INDEX); 1570fa4335d7SJohan Hedberg goto done; 1571fa4335d7SJohan Hedberg } 1572fa4335d7SJohan Hedberg 1573fa4335d7SJohan Hedberg if (hci_dev_test_flag(hdev, HCI_UNCONFIGURED) && 1574fa4335d7SJohan Hedberg !(handler->flags & HCI_MGMT_UNCONFIGURED)) { 1575fa4335d7SJohan Hedberg err = mgmt_cmd_status(sk, index, opcode, 1576fa4335d7SJohan Hedberg MGMT_STATUS_INVALID_INDEX); 1577fa4335d7SJohan Hedberg goto done; 1578fa4335d7SJohan Hedberg } 1579fa4335d7SJohan Hedberg } 1580fa4335d7SJohan Hedberg 1581d5cc6626SMarcel Holtmann if (!(handler->flags & HCI_MGMT_HDEV_OPTIONAL)) { 1582fa4335d7SJohan Hedberg no_hdev = (handler->flags & HCI_MGMT_NO_HDEV); 1583fa4335d7SJohan Hedberg if (no_hdev != !hdev) { 1584fa4335d7SJohan Hedberg err = mgmt_cmd_status(sk, index, opcode, 1585fa4335d7SJohan Hedberg MGMT_STATUS_INVALID_INDEX); 1586fa4335d7SJohan Hedberg goto done; 1587fa4335d7SJohan Hedberg } 1588d5cc6626SMarcel Holtmann } 1589fa4335d7SJohan Hedberg 1590fa4335d7SJohan Hedberg var_len = (handler->flags & HCI_MGMT_VAR_LEN); 1591fa4335d7SJohan Hedberg if ((var_len && len < handler->data_len) || 1592fa4335d7SJohan Hedberg (!var_len && len != handler->data_len)) { 1593fa4335d7SJohan Hedberg err = mgmt_cmd_status(sk, index, opcode, 1594fa4335d7SJohan Hedberg MGMT_STATUS_INVALID_PARAMS); 1595fa4335d7SJohan Hedberg goto done; 1596fa4335d7SJohan Hedberg } 1597fa4335d7SJohan Hedberg 1598fa4335d7SJohan Hedberg if (hdev && chan->hdev_init) 1599fa4335d7SJohan Hedberg chan->hdev_init(sk, hdev); 1600fa4335d7SJohan Hedberg 1601fa4335d7SJohan Hedberg cp = buf + sizeof(*hdr); 1602fa4335d7SJohan Hedberg 1603fa4335d7SJohan Hedberg err = handler->func(sk, hdev, cp, len); 1604fa4335d7SJohan Hedberg if (err < 0) 1605fa4335d7SJohan Hedberg goto done; 1606fa4335d7SJohan Hedberg 1607fa4335d7SJohan Hedberg err = msglen; 1608fa4335d7SJohan Hedberg 1609fa4335d7SJohan Hedberg done: 1610fa4335d7SJohan Hedberg if (hdev) 1611fa4335d7SJohan Hedberg hci_dev_put(hdev); 1612fa4335d7SJohan Hedberg 1613fa4335d7SJohan Hedberg kfree(buf); 1614fa4335d7SJohan Hedberg return err; 1615fa4335d7SJohan Hedberg } 1616fa4335d7SJohan Hedberg 1617ac714949SMarcel Holtmann static int hci_logging_frame(struct sock *sk, struct msghdr *msg, int len) 1618ac714949SMarcel Holtmann { 1619ac714949SMarcel Holtmann struct hci_mon_hdr *hdr; 1620ac714949SMarcel Holtmann struct sk_buff *skb; 1621ac714949SMarcel Holtmann struct hci_dev *hdev; 1622ac714949SMarcel Holtmann u16 index; 1623ac714949SMarcel Holtmann int err; 1624ac714949SMarcel Holtmann 1625ac714949SMarcel Holtmann /* The logging frame consists at minimum of the standard header, 1626ac714949SMarcel Holtmann * the priority byte, the ident length byte and at least one string 1627ac714949SMarcel Holtmann * terminator NUL byte. Anything shorter are invalid packets. 1628ac714949SMarcel Holtmann */ 1629ac714949SMarcel Holtmann if (len < sizeof(*hdr) + 3) 1630ac714949SMarcel Holtmann return -EINVAL; 1631ac714949SMarcel Holtmann 1632ac714949SMarcel Holtmann skb = bt_skb_send_alloc(sk, len, msg->msg_flags & MSG_DONTWAIT, &err); 1633ac714949SMarcel Holtmann if (!skb) 1634ac714949SMarcel Holtmann return err; 1635ac714949SMarcel Holtmann 1636ac714949SMarcel Holtmann if (memcpy_from_msg(skb_put(skb, len), msg, len)) { 1637ac714949SMarcel Holtmann err = -EFAULT; 1638ac714949SMarcel Holtmann goto drop; 1639ac714949SMarcel Holtmann } 1640ac714949SMarcel Holtmann 1641ac714949SMarcel Holtmann hdr = (void *)skb->data; 1642ac714949SMarcel Holtmann 1643ac714949SMarcel Holtmann if (__le16_to_cpu(hdr->len) != len - sizeof(*hdr)) { 1644ac714949SMarcel Holtmann err = -EINVAL; 1645ac714949SMarcel Holtmann goto drop; 1646ac714949SMarcel Holtmann } 1647ac714949SMarcel Holtmann 1648ac714949SMarcel Holtmann if (__le16_to_cpu(hdr->opcode) == 0x0000) { 1649ac714949SMarcel Holtmann __u8 priority = skb->data[sizeof(*hdr)]; 1650ac714949SMarcel Holtmann __u8 ident_len = skb->data[sizeof(*hdr) + 1]; 1651ac714949SMarcel Holtmann 1652ac714949SMarcel Holtmann /* Only the priorities 0-7 are valid and with that any other 1653ac714949SMarcel Holtmann * value results in an invalid packet. 1654ac714949SMarcel Holtmann * 1655ac714949SMarcel Holtmann * The priority byte is followed by an ident length byte and 1656ac714949SMarcel Holtmann * the NUL terminated ident string. Check that the ident 1657ac714949SMarcel Holtmann * length is not overflowing the packet and also that the 1658ac714949SMarcel Holtmann * ident string itself is NUL terminated. In case the ident 1659ac714949SMarcel Holtmann * length is zero, the length value actually doubles as NUL 1660ac714949SMarcel Holtmann * terminator identifier. 1661ac714949SMarcel Holtmann * 1662ac714949SMarcel Holtmann * The message follows the ident string (if present) and 1663ac714949SMarcel Holtmann * must be NUL terminated. Otherwise it is not a valid packet. 1664ac714949SMarcel Holtmann */ 1665ac714949SMarcel Holtmann if (priority > 7 || skb->data[len - 1] != 0x00 || 1666ac714949SMarcel Holtmann ident_len > len - sizeof(*hdr) - 3 || 1667ac714949SMarcel Holtmann skb->data[sizeof(*hdr) + ident_len + 1] != 0x00) { 1668ac714949SMarcel Holtmann err = -EINVAL; 1669ac714949SMarcel Holtmann goto drop; 1670ac714949SMarcel Holtmann } 1671ac714949SMarcel Holtmann } else { 1672ac714949SMarcel Holtmann err = -EINVAL; 1673ac714949SMarcel Holtmann goto drop; 1674ac714949SMarcel Holtmann } 1675ac714949SMarcel Holtmann 1676ac714949SMarcel Holtmann index = __le16_to_cpu(hdr->index); 1677ac714949SMarcel Holtmann 1678ac714949SMarcel Holtmann if (index != MGMT_INDEX_NONE) { 1679ac714949SMarcel Holtmann hdev = hci_dev_get(index); 1680ac714949SMarcel Holtmann if (!hdev) { 1681ac714949SMarcel Holtmann err = -ENODEV; 1682ac714949SMarcel Holtmann goto drop; 1683ac714949SMarcel Holtmann } 1684ac714949SMarcel Holtmann } else { 1685ac714949SMarcel Holtmann hdev = NULL; 1686ac714949SMarcel Holtmann } 1687ac714949SMarcel Holtmann 1688ac714949SMarcel Holtmann hdr->opcode = cpu_to_le16(HCI_MON_USER_LOGGING); 1689ac714949SMarcel Holtmann 1690ac714949SMarcel Holtmann hci_send_to_channel(HCI_CHANNEL_MONITOR, skb, HCI_SOCK_TRUSTED, NULL); 1691ac714949SMarcel Holtmann err = len; 1692ac714949SMarcel Holtmann 1693ac714949SMarcel Holtmann if (hdev) 1694ac714949SMarcel Holtmann hci_dev_put(hdev); 1695ac714949SMarcel Holtmann 1696ac714949SMarcel Holtmann drop: 1697ac714949SMarcel Holtmann kfree_skb(skb); 1698ac714949SMarcel Holtmann return err; 1699ac714949SMarcel Holtmann } 1700ac714949SMarcel Holtmann 17011b784140SYing Xue static int hci_sock_sendmsg(struct socket *sock, struct msghdr *msg, 17021b784140SYing Xue size_t len) 17031da177e4SLinus Torvalds { 17041da177e4SLinus Torvalds struct sock *sk = sock->sk; 1705801c1e8dSJohan Hedberg struct hci_mgmt_chan *chan; 17061da177e4SLinus Torvalds struct hci_dev *hdev; 17071da177e4SLinus Torvalds struct sk_buff *skb; 17081da177e4SLinus Torvalds int err; 17091da177e4SLinus Torvalds 17101da177e4SLinus Torvalds BT_DBG("sock %p sk %p", sock, sk); 17111da177e4SLinus Torvalds 17121da177e4SLinus Torvalds if (msg->msg_flags & MSG_OOB) 17131da177e4SLinus Torvalds return -EOPNOTSUPP; 17141da177e4SLinus Torvalds 1715ab89f0bdSSzymon Janc if (msg->msg_flags & ~(MSG_DONTWAIT|MSG_NOSIGNAL|MSG_ERRQUEUE| 1716ab89f0bdSSzymon Janc MSG_CMSG_COMPAT)) 17171da177e4SLinus Torvalds return -EINVAL; 17181da177e4SLinus Torvalds 17191da177e4SLinus Torvalds if (len < 4 || len > HCI_MAX_FRAME_SIZE) 17201da177e4SLinus Torvalds return -EINVAL; 17211da177e4SLinus Torvalds 17221da177e4SLinus Torvalds lock_sock(sk); 17231da177e4SLinus Torvalds 17240381101fSJohan Hedberg switch (hci_pi(sk)->channel) { 17250381101fSJohan Hedberg case HCI_CHANNEL_RAW: 172623500189SMarcel Holtmann case HCI_CHANNEL_USER: 17270381101fSJohan Hedberg break; 1728cd82e61cSMarcel Holtmann case HCI_CHANNEL_MONITOR: 1729cd82e61cSMarcel Holtmann err = -EOPNOTSUPP; 1730cd82e61cSMarcel Holtmann goto done; 1731ac714949SMarcel Holtmann case HCI_CHANNEL_LOGGING: 1732ac714949SMarcel Holtmann err = hci_logging_frame(sk, msg, len); 1733ac714949SMarcel Holtmann goto done; 17340381101fSJohan Hedberg default: 1735801c1e8dSJohan Hedberg mutex_lock(&mgmt_chan_list_lock); 1736801c1e8dSJohan Hedberg chan = __hci_mgmt_chan_find(hci_pi(sk)->channel); 1737801c1e8dSJohan Hedberg if (chan) 1738fa4335d7SJohan Hedberg err = hci_mgmt_cmd(chan, sk, msg, len); 1739801c1e8dSJohan Hedberg else 17400381101fSJohan Hedberg err = -EINVAL; 1741801c1e8dSJohan Hedberg 1742801c1e8dSJohan Hedberg mutex_unlock(&mgmt_chan_list_lock); 17430381101fSJohan Hedberg goto done; 17440381101fSJohan Hedberg } 17450381101fSJohan Hedberg 174670f23020SAndrei Emeltchenko hdev = hci_pi(sk)->hdev; 174770f23020SAndrei Emeltchenko if (!hdev) { 17481da177e4SLinus Torvalds err = -EBADFD; 17491da177e4SLinus Torvalds goto done; 17501da177e4SLinus Torvalds } 17511da177e4SLinus Torvalds 17527e21addcSMarcel Holtmann if (!test_bit(HCI_UP, &hdev->flags)) { 17537e21addcSMarcel Holtmann err = -ENETDOWN; 17547e21addcSMarcel Holtmann goto done; 17557e21addcSMarcel Holtmann } 17567e21addcSMarcel Holtmann 175770f23020SAndrei Emeltchenko skb = bt_skb_send_alloc(sk, len, msg->msg_flags & MSG_DONTWAIT, &err); 175870f23020SAndrei Emeltchenko if (!skb) 17591da177e4SLinus Torvalds goto done; 17601da177e4SLinus Torvalds 17616ce8e9ceSAl Viro if (memcpy_from_msg(skb_put(skb, len), msg, len)) { 17621da177e4SLinus Torvalds err = -EFAULT; 17631da177e4SLinus Torvalds goto drop; 17641da177e4SLinus Torvalds } 17651da177e4SLinus Torvalds 17668528d3f7SMarcel Holtmann hci_skb_pkt_type(skb) = skb->data[0]; 17671da177e4SLinus Torvalds skb_pull(skb, 1); 17681da177e4SLinus Torvalds 17691bc5ad16SMarcel Holtmann if (hci_pi(sk)->channel == HCI_CHANNEL_USER) { 17701bc5ad16SMarcel Holtmann /* No permission check is needed for user channel 17711bc5ad16SMarcel Holtmann * since that gets enforced when binding the socket. 17721bc5ad16SMarcel Holtmann * 17731bc5ad16SMarcel Holtmann * However check that the packet type is valid. 17741bc5ad16SMarcel Holtmann */ 1775d79f34e3SMarcel Holtmann if (hci_skb_pkt_type(skb) != HCI_COMMAND_PKT && 1776d79f34e3SMarcel Holtmann hci_skb_pkt_type(skb) != HCI_ACLDATA_PKT && 1777cc974003SMarcel Holtmann hci_skb_pkt_type(skb) != HCI_SCODATA_PKT && 1778cc974003SMarcel Holtmann hci_skb_pkt_type(skb) != HCI_ISODATA_PKT) { 17791bc5ad16SMarcel Holtmann err = -EINVAL; 17801bc5ad16SMarcel Holtmann goto drop; 17811bc5ad16SMarcel Holtmann } 17821bc5ad16SMarcel Holtmann 17831bc5ad16SMarcel Holtmann skb_queue_tail(&hdev->raw_q, skb); 17841bc5ad16SMarcel Holtmann queue_work(hdev->workqueue, &hdev->tx_work); 1785d79f34e3SMarcel Holtmann } else if (hci_skb_pkt_type(skb) == HCI_COMMAND_PKT) { 178683985319SHarvey Harrison u16 opcode = get_unaligned_le16(skb->data); 17871da177e4SLinus Torvalds u16 ogf = hci_opcode_ogf(opcode); 17881da177e4SLinus Torvalds u16 ocf = hci_opcode_ocf(opcode); 17891da177e4SLinus Torvalds 17901da177e4SLinus Torvalds if (((ogf > HCI_SFLT_MAX_OGF) || 17913bb3c755SGustavo Padovan !hci_test_bit(ocf & HCI_FLT_OCF_BITS, 17923bb3c755SGustavo Padovan &hci_sec_filter.ocf_mask[ogf])) && 17931da177e4SLinus Torvalds !capable(CAP_NET_RAW)) { 17941da177e4SLinus Torvalds err = -EPERM; 17951da177e4SLinus Torvalds goto drop; 17961da177e4SLinus Torvalds } 17971da177e4SLinus Torvalds 17981982162bSMarcel Holtmann /* Since the opcode has already been extracted here, store 17991982162bSMarcel Holtmann * a copy of the value for later use by the drivers. 18001982162bSMarcel Holtmann */ 18011982162bSMarcel Holtmann hci_skb_opcode(skb) = opcode; 18021982162bSMarcel Holtmann 1803fee746b0SMarcel Holtmann if (ogf == 0x3f) { 18041da177e4SLinus Torvalds skb_queue_tail(&hdev->raw_q, skb); 18053eff45eaSGustavo F. Padovan queue_work(hdev->workqueue, &hdev->tx_work); 18061da177e4SLinus Torvalds } else { 180749c922bbSStephen Hemminger /* Stand-alone HCI commands must be flagged as 180811714b3dSJohan Hedberg * single-command requests. 180911714b3dSJohan Hedberg */ 181044d27137SJohan Hedberg bt_cb(skb)->hci.req_flags |= HCI_REQ_START; 181111714b3dSJohan Hedberg 18121da177e4SLinus Torvalds skb_queue_tail(&hdev->cmd_q, skb); 1813c347b765SGustavo F. Padovan queue_work(hdev->workqueue, &hdev->cmd_work); 18141da177e4SLinus Torvalds } 18151da177e4SLinus Torvalds } else { 18161da177e4SLinus Torvalds if (!capable(CAP_NET_RAW)) { 18171da177e4SLinus Torvalds err = -EPERM; 18181da177e4SLinus Torvalds goto drop; 18191da177e4SLinus Torvalds } 18201da177e4SLinus Torvalds 1821d79f34e3SMarcel Holtmann if (hci_skb_pkt_type(skb) != HCI_ACLDATA_PKT && 1822cc974003SMarcel Holtmann hci_skb_pkt_type(skb) != HCI_SCODATA_PKT && 1823cc974003SMarcel Holtmann hci_skb_pkt_type(skb) != HCI_ISODATA_PKT) { 1824bb77543eSMarcel Holtmann err = -EINVAL; 1825bb77543eSMarcel Holtmann goto drop; 1826bb77543eSMarcel Holtmann } 1827bb77543eSMarcel Holtmann 18281da177e4SLinus Torvalds skb_queue_tail(&hdev->raw_q, skb); 18293eff45eaSGustavo F. Padovan queue_work(hdev->workqueue, &hdev->tx_work); 18301da177e4SLinus Torvalds } 18311da177e4SLinus Torvalds 18321da177e4SLinus Torvalds err = len; 18331da177e4SLinus Torvalds 18341da177e4SLinus Torvalds done: 18351da177e4SLinus Torvalds release_sock(sk); 18361da177e4SLinus Torvalds return err; 18371da177e4SLinus Torvalds 18381da177e4SLinus Torvalds drop: 18391da177e4SLinus Torvalds kfree_skb(skb); 18401da177e4SLinus Torvalds goto done; 18411da177e4SLinus Torvalds } 18421da177e4SLinus Torvalds 18438fc9ced3SGustavo Padovan static int hci_sock_setsockopt(struct socket *sock, int level, int optname, 1844a7b75c5aSChristoph Hellwig sockptr_t optval, unsigned int len) 18451da177e4SLinus Torvalds { 18461da177e4SLinus Torvalds struct hci_ufilter uf = { .opcode = 0 }; 18471da177e4SLinus Torvalds struct sock *sk = sock->sk; 18481da177e4SLinus Torvalds int err = 0, opt = 0; 18491da177e4SLinus Torvalds 18501da177e4SLinus Torvalds BT_DBG("sk %p, opt %d", sk, optname); 18511da177e4SLinus Torvalds 185247b0f573SMarcel Holtmann if (level != SOL_HCI) 185347b0f573SMarcel Holtmann return -ENOPROTOOPT; 185447b0f573SMarcel Holtmann 18551da177e4SLinus Torvalds lock_sock(sk); 18561da177e4SLinus Torvalds 18572f39cdb7SMarcel Holtmann if (hci_pi(sk)->channel != HCI_CHANNEL_RAW) { 1858c2371e80SMarcel Holtmann err = -EBADFD; 18592f39cdb7SMarcel Holtmann goto done; 18602f39cdb7SMarcel Holtmann } 18612f39cdb7SMarcel Holtmann 18621da177e4SLinus Torvalds switch (optname) { 18631da177e4SLinus Torvalds case HCI_DATA_DIR: 1864a7b75c5aSChristoph Hellwig if (copy_from_sockptr(&opt, optval, sizeof(opt))) { 18651da177e4SLinus Torvalds err = -EFAULT; 18661da177e4SLinus Torvalds break; 18671da177e4SLinus Torvalds } 18681da177e4SLinus Torvalds 18691da177e4SLinus Torvalds if (opt) 18701da177e4SLinus Torvalds hci_pi(sk)->cmsg_mask |= HCI_CMSG_DIR; 18711da177e4SLinus Torvalds else 18721da177e4SLinus Torvalds hci_pi(sk)->cmsg_mask &= ~HCI_CMSG_DIR; 18731da177e4SLinus Torvalds break; 18741da177e4SLinus Torvalds 18751da177e4SLinus Torvalds case HCI_TIME_STAMP: 1876a7b75c5aSChristoph Hellwig if (copy_from_sockptr(&opt, optval, sizeof(opt))) { 18771da177e4SLinus Torvalds err = -EFAULT; 18781da177e4SLinus Torvalds break; 18791da177e4SLinus Torvalds } 18801da177e4SLinus Torvalds 18811da177e4SLinus Torvalds if (opt) 18821da177e4SLinus Torvalds hci_pi(sk)->cmsg_mask |= HCI_CMSG_TSTAMP; 18831da177e4SLinus Torvalds else 18841da177e4SLinus Torvalds hci_pi(sk)->cmsg_mask &= ~HCI_CMSG_TSTAMP; 18851da177e4SLinus Torvalds break; 18861da177e4SLinus Torvalds 18871da177e4SLinus Torvalds case HCI_FILTER: 18880878b666SMarcel Holtmann { 18890878b666SMarcel Holtmann struct hci_filter *f = &hci_pi(sk)->filter; 18900878b666SMarcel Holtmann 18910878b666SMarcel Holtmann uf.type_mask = f->type_mask; 18920878b666SMarcel Holtmann uf.opcode = f->opcode; 18930878b666SMarcel Holtmann uf.event_mask[0] = *((u32 *) f->event_mask + 0); 18940878b666SMarcel Holtmann uf.event_mask[1] = *((u32 *) f->event_mask + 1); 18950878b666SMarcel Holtmann } 18960878b666SMarcel Holtmann 18971da177e4SLinus Torvalds len = min_t(unsigned int, len, sizeof(uf)); 1898a7b75c5aSChristoph Hellwig if (copy_from_sockptr(&uf, optval, len)) { 18991da177e4SLinus Torvalds err = -EFAULT; 19001da177e4SLinus Torvalds break; 19011da177e4SLinus Torvalds } 19021da177e4SLinus Torvalds 19031da177e4SLinus Torvalds if (!capable(CAP_NET_RAW)) { 19041da177e4SLinus Torvalds uf.type_mask &= hci_sec_filter.type_mask; 19051da177e4SLinus Torvalds uf.event_mask[0] &= *((u32 *) hci_sec_filter.event_mask + 0); 19061da177e4SLinus Torvalds uf.event_mask[1] &= *((u32 *) hci_sec_filter.event_mask + 1); 19071da177e4SLinus Torvalds } 19081da177e4SLinus Torvalds 19091da177e4SLinus Torvalds { 19101da177e4SLinus Torvalds struct hci_filter *f = &hci_pi(sk)->filter; 19111da177e4SLinus Torvalds 19121da177e4SLinus Torvalds f->type_mask = uf.type_mask; 19131da177e4SLinus Torvalds f->opcode = uf.opcode; 19141da177e4SLinus Torvalds *((u32 *) f->event_mask + 0) = uf.event_mask[0]; 19151da177e4SLinus Torvalds *((u32 *) f->event_mask + 1) = uf.event_mask[1]; 19161da177e4SLinus Torvalds } 19171da177e4SLinus Torvalds break; 19181da177e4SLinus Torvalds 19191da177e4SLinus Torvalds default: 19201da177e4SLinus Torvalds err = -ENOPROTOOPT; 19211da177e4SLinus Torvalds break; 19221da177e4SLinus Torvalds } 19231da177e4SLinus Torvalds 19242f39cdb7SMarcel Holtmann done: 19251da177e4SLinus Torvalds release_sock(sk); 19261da177e4SLinus Torvalds return err; 19271da177e4SLinus Torvalds } 19281da177e4SLinus Torvalds 19298fc9ced3SGustavo Padovan static int hci_sock_getsockopt(struct socket *sock, int level, int optname, 19308fc9ced3SGustavo Padovan char __user *optval, int __user *optlen) 19311da177e4SLinus Torvalds { 19321da177e4SLinus Torvalds struct hci_ufilter uf; 19331da177e4SLinus Torvalds struct sock *sk = sock->sk; 1934cedc5469SMarcel Holtmann int len, opt, err = 0; 1935cedc5469SMarcel Holtmann 1936cedc5469SMarcel Holtmann BT_DBG("sk %p, opt %d", sk, optname); 19371da177e4SLinus Torvalds 193847b0f573SMarcel Holtmann if (level != SOL_HCI) 193947b0f573SMarcel Holtmann return -ENOPROTOOPT; 194047b0f573SMarcel Holtmann 19411da177e4SLinus Torvalds if (get_user(len, optlen)) 19421da177e4SLinus Torvalds return -EFAULT; 19431da177e4SLinus Torvalds 1944cedc5469SMarcel Holtmann lock_sock(sk); 1945cedc5469SMarcel Holtmann 1946cedc5469SMarcel Holtmann if (hci_pi(sk)->channel != HCI_CHANNEL_RAW) { 1947c2371e80SMarcel Holtmann err = -EBADFD; 1948cedc5469SMarcel Holtmann goto done; 1949cedc5469SMarcel Holtmann } 1950cedc5469SMarcel Holtmann 19511da177e4SLinus Torvalds switch (optname) { 19521da177e4SLinus Torvalds case HCI_DATA_DIR: 19531da177e4SLinus Torvalds if (hci_pi(sk)->cmsg_mask & HCI_CMSG_DIR) 19541da177e4SLinus Torvalds opt = 1; 19551da177e4SLinus Torvalds else 19561da177e4SLinus Torvalds opt = 0; 19571da177e4SLinus Torvalds 19581da177e4SLinus Torvalds if (put_user(opt, optval)) 1959cedc5469SMarcel Holtmann err = -EFAULT; 19601da177e4SLinus Torvalds break; 19611da177e4SLinus Torvalds 19621da177e4SLinus Torvalds case HCI_TIME_STAMP: 19631da177e4SLinus Torvalds if (hci_pi(sk)->cmsg_mask & HCI_CMSG_TSTAMP) 19641da177e4SLinus Torvalds opt = 1; 19651da177e4SLinus Torvalds else 19661da177e4SLinus Torvalds opt = 0; 19671da177e4SLinus Torvalds 19681da177e4SLinus Torvalds if (put_user(opt, optval)) 1969cedc5469SMarcel Holtmann err = -EFAULT; 19701da177e4SLinus Torvalds break; 19711da177e4SLinus Torvalds 19721da177e4SLinus Torvalds case HCI_FILTER: 19731da177e4SLinus Torvalds { 19741da177e4SLinus Torvalds struct hci_filter *f = &hci_pi(sk)->filter; 19751da177e4SLinus Torvalds 1976e15ca9a0SMathias Krause memset(&uf, 0, sizeof(uf)); 19771da177e4SLinus Torvalds uf.type_mask = f->type_mask; 19781da177e4SLinus Torvalds uf.opcode = f->opcode; 19791da177e4SLinus Torvalds uf.event_mask[0] = *((u32 *) f->event_mask + 0); 19801da177e4SLinus Torvalds uf.event_mask[1] = *((u32 *) f->event_mask + 1); 19811da177e4SLinus Torvalds } 19821da177e4SLinus Torvalds 19831da177e4SLinus Torvalds len = min_t(unsigned int, len, sizeof(uf)); 19841da177e4SLinus Torvalds if (copy_to_user(optval, &uf, len)) 1985cedc5469SMarcel Holtmann err = -EFAULT; 19861da177e4SLinus Torvalds break; 19871da177e4SLinus Torvalds 19881da177e4SLinus Torvalds default: 1989cedc5469SMarcel Holtmann err = -ENOPROTOOPT; 19901da177e4SLinus Torvalds break; 19911da177e4SLinus Torvalds } 19921da177e4SLinus Torvalds 1993cedc5469SMarcel Holtmann done: 1994cedc5469SMarcel Holtmann release_sock(sk); 1995cedc5469SMarcel Holtmann return err; 19961da177e4SLinus Torvalds } 19971da177e4SLinus Torvalds 199890ddc4f0SEric Dumazet static const struct proto_ops hci_sock_ops = { 19991da177e4SLinus Torvalds .family = PF_BLUETOOTH, 20001da177e4SLinus Torvalds .owner = THIS_MODULE, 20011da177e4SLinus Torvalds .release = hci_sock_release, 20021da177e4SLinus Torvalds .bind = hci_sock_bind, 20031da177e4SLinus Torvalds .getname = hci_sock_getname, 20041da177e4SLinus Torvalds .sendmsg = hci_sock_sendmsg, 20051da177e4SLinus Torvalds .recvmsg = hci_sock_recvmsg, 20061da177e4SLinus Torvalds .ioctl = hci_sock_ioctl, 20077a6038b3SArnd Bergmann #ifdef CONFIG_COMPAT 20087a6038b3SArnd Bergmann .compat_ioctl = hci_sock_compat_ioctl, 20097a6038b3SArnd Bergmann #endif 2010a11e1d43SLinus Torvalds .poll = datagram_poll, 20111da177e4SLinus Torvalds .listen = sock_no_listen, 20121da177e4SLinus Torvalds .shutdown = sock_no_shutdown, 20131da177e4SLinus Torvalds .setsockopt = hci_sock_setsockopt, 20141da177e4SLinus Torvalds .getsockopt = hci_sock_getsockopt, 20151da177e4SLinus Torvalds .connect = sock_no_connect, 20161da177e4SLinus Torvalds .socketpair = sock_no_socketpair, 20171da177e4SLinus Torvalds .accept = sock_no_accept, 20181da177e4SLinus Torvalds .mmap = sock_no_mmap 20191da177e4SLinus Torvalds }; 20201da177e4SLinus Torvalds 20211da177e4SLinus Torvalds static struct proto hci_sk_proto = { 20221da177e4SLinus Torvalds .name = "HCI", 20231da177e4SLinus Torvalds .owner = THIS_MODULE, 20241da177e4SLinus Torvalds .obj_size = sizeof(struct hci_pinfo) 20251da177e4SLinus Torvalds }; 20261da177e4SLinus Torvalds 20273f378b68SEric Paris static int hci_sock_create(struct net *net, struct socket *sock, int protocol, 20283f378b68SEric Paris int kern) 20291da177e4SLinus Torvalds { 20301da177e4SLinus Torvalds struct sock *sk; 20311da177e4SLinus Torvalds 20321da177e4SLinus Torvalds BT_DBG("sock %p", sock); 20331da177e4SLinus Torvalds 20341da177e4SLinus Torvalds if (sock->type != SOCK_RAW) 20351da177e4SLinus Torvalds return -ESOCKTNOSUPPORT; 20361da177e4SLinus Torvalds 20371da177e4SLinus Torvalds sock->ops = &hci_sock_ops; 20381da177e4SLinus Torvalds 203911aa9c28SEric W. Biederman sk = sk_alloc(net, PF_BLUETOOTH, GFP_ATOMIC, &hci_sk_proto, kern); 20401da177e4SLinus Torvalds if (!sk) 20411da177e4SLinus Torvalds return -ENOMEM; 20421da177e4SLinus Torvalds 20431da177e4SLinus Torvalds sock_init_data(sock, sk); 20441da177e4SLinus Torvalds 20451da177e4SLinus Torvalds sock_reset_flag(sk, SOCK_ZAPPED); 20461da177e4SLinus Torvalds 20471da177e4SLinus Torvalds sk->sk_protocol = protocol; 20481da177e4SLinus Torvalds 20491da177e4SLinus Torvalds sock->state = SS_UNCONNECTED; 20501da177e4SLinus Torvalds sk->sk_state = BT_OPEN; 20511da177e4SLinus Torvalds 20521da177e4SLinus Torvalds bt_sock_link(&hci_sk_list, sk); 20531da177e4SLinus Torvalds return 0; 20541da177e4SLinus Torvalds } 20551da177e4SLinus Torvalds 2056ec1b4cf7SStephen Hemminger static const struct net_proto_family hci_sock_family_ops = { 20571da177e4SLinus Torvalds .family = PF_BLUETOOTH, 20581da177e4SLinus Torvalds .owner = THIS_MODULE, 20591da177e4SLinus Torvalds .create = hci_sock_create, 20601da177e4SLinus Torvalds }; 20611da177e4SLinus Torvalds 20621da177e4SLinus Torvalds int __init hci_sock_init(void) 20631da177e4SLinus Torvalds { 20641da177e4SLinus Torvalds int err; 20651da177e4SLinus Torvalds 2066b0a8e282SMarcel Holtmann BUILD_BUG_ON(sizeof(struct sockaddr_hci) > sizeof(struct sockaddr)); 2067b0a8e282SMarcel Holtmann 20681da177e4SLinus Torvalds err = proto_register(&hci_sk_proto, 0); 20691da177e4SLinus Torvalds if (err < 0) 20701da177e4SLinus Torvalds return err; 20711da177e4SLinus Torvalds 20721da177e4SLinus Torvalds err = bt_sock_register(BTPROTO_HCI, &hci_sock_family_ops); 2073f7c86637SMasatake YAMATO if (err < 0) { 2074f7c86637SMasatake YAMATO BT_ERR("HCI socket registration failed"); 20751da177e4SLinus Torvalds goto error; 2076f7c86637SMasatake YAMATO } 2077f7c86637SMasatake YAMATO 2078b0316615SAl Viro err = bt_procfs_init(&init_net, "hci", &hci_sk_list, NULL); 2079f7c86637SMasatake YAMATO if (err < 0) { 2080f7c86637SMasatake YAMATO BT_ERR("Failed to create HCI proc file"); 2081f7c86637SMasatake YAMATO bt_sock_unregister(BTPROTO_HCI); 2082f7c86637SMasatake YAMATO goto error; 2083f7c86637SMasatake YAMATO } 20841da177e4SLinus Torvalds 20851da177e4SLinus Torvalds BT_INFO("HCI socket layer initialized"); 20861da177e4SLinus Torvalds 20871da177e4SLinus Torvalds return 0; 20881da177e4SLinus Torvalds 20891da177e4SLinus Torvalds error: 20901da177e4SLinus Torvalds proto_unregister(&hci_sk_proto); 20911da177e4SLinus Torvalds return err; 20921da177e4SLinus Torvalds } 20931da177e4SLinus Torvalds 2094b7440a14SAnand Gadiyar void hci_sock_cleanup(void) 20951da177e4SLinus Torvalds { 2096f7c86637SMasatake YAMATO bt_procfs_cleanup(&init_net, "hci"); 20975e9d7f86SDavid Herrmann bt_sock_unregister(BTPROTO_HCI); 20981da177e4SLinus Torvalds proto_unregister(&hci_sk_proto); 20991da177e4SLinus Torvalds } 2100