xref: /openbmc/linux/net/bluetooth/hci_sock.c (revision 0b59e272f9324dac8c12444cf5926be84abd53f9)
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 
62e0448092STetsuo Handa static struct hci_dev *hci_hdev_from_sock(struct sock *sk)
63e0448092STetsuo Handa {
64e0448092STetsuo Handa 	struct hci_dev *hdev = hci_pi(sk)->hdev;
65e0448092STetsuo Handa 
66e0448092STetsuo Handa 	if (!hdev)
67e0448092STetsuo Handa 		return ERR_PTR(-EBADFD);
68e0448092STetsuo Handa 	if (hci_dev_test_flag(hdev, HCI_UNREGISTER))
69e0448092STetsuo Handa 		return ERR_PTR(-EPIPE);
70e0448092STetsuo Handa 	return hdev;
71e0448092STetsuo Handa }
72e0448092STetsuo Handa 
736befc644SMarcel Holtmann void hci_sock_set_flag(struct sock *sk, int nr)
746befc644SMarcel Holtmann {
756befc644SMarcel Holtmann 	set_bit(nr, &hci_pi(sk)->flags);
766befc644SMarcel Holtmann }
776befc644SMarcel Holtmann 
786befc644SMarcel Holtmann void hci_sock_clear_flag(struct sock *sk, int nr)
796befc644SMarcel Holtmann {
806befc644SMarcel Holtmann 	clear_bit(nr, &hci_pi(sk)->flags);
816befc644SMarcel Holtmann }
826befc644SMarcel Holtmann 
83c85be545SMarcel Holtmann int hci_sock_test_flag(struct sock *sk, int nr)
84c85be545SMarcel Holtmann {
85c85be545SMarcel Holtmann 	return test_bit(nr, &hci_pi(sk)->flags);
86c85be545SMarcel Holtmann }
87c85be545SMarcel Holtmann 
88d0f172b1SJohan Hedberg unsigned short hci_sock_get_channel(struct sock *sk)
89d0f172b1SJohan Hedberg {
90d0f172b1SJohan Hedberg 	return hci_pi(sk)->channel;
91d0f172b1SJohan Hedberg }
92d0f172b1SJohan Hedberg 
9370ecce91SMarcel Holtmann u32 hci_sock_get_cookie(struct sock *sk)
9470ecce91SMarcel Holtmann {
9570ecce91SMarcel Holtmann 	return hci_pi(sk)->cookie;
9670ecce91SMarcel Holtmann }
9770ecce91SMarcel Holtmann 
98df1cb87aSMarcel Holtmann static bool hci_sock_gen_cookie(struct sock *sk)
99df1cb87aSMarcel Holtmann {
100df1cb87aSMarcel Holtmann 	int id = hci_pi(sk)->cookie;
101df1cb87aSMarcel Holtmann 
102df1cb87aSMarcel Holtmann 	if (!id) {
103df1cb87aSMarcel Holtmann 		id = ida_simple_get(&sock_cookie_ida, 1, 0, GFP_KERNEL);
104df1cb87aSMarcel Holtmann 		if (id < 0)
105df1cb87aSMarcel Holtmann 			id = 0xffffffff;
106df1cb87aSMarcel Holtmann 
107df1cb87aSMarcel Holtmann 		hci_pi(sk)->cookie = id;
108df1cb87aSMarcel Holtmann 		get_task_comm(hci_pi(sk)->comm, current);
109df1cb87aSMarcel Holtmann 		return true;
110df1cb87aSMarcel Holtmann 	}
111df1cb87aSMarcel Holtmann 
112df1cb87aSMarcel Holtmann 	return false;
113df1cb87aSMarcel Holtmann }
114df1cb87aSMarcel Holtmann 
115df1cb87aSMarcel Holtmann static void hci_sock_free_cookie(struct sock *sk)
116df1cb87aSMarcel Holtmann {
117df1cb87aSMarcel Holtmann 	int id = hci_pi(sk)->cookie;
118df1cb87aSMarcel Holtmann 
119df1cb87aSMarcel Holtmann 	if (id) {
120df1cb87aSMarcel Holtmann 		hci_pi(sk)->cookie = 0xffffffff;
121df1cb87aSMarcel Holtmann 		ida_simple_remove(&sock_cookie_ida, id);
122df1cb87aSMarcel Holtmann 	}
123df1cb87aSMarcel Holtmann }
124df1cb87aSMarcel Holtmann 
1259391976aSJiri Slaby static inline int hci_test_bit(int nr, const void *addr)
1261da177e4SLinus Torvalds {
1279391976aSJiri Slaby 	return *((const __u32 *) addr + (nr >> 5)) & ((__u32) 1 << (nr & 31));
1281da177e4SLinus Torvalds }
1291da177e4SLinus Torvalds 
1301da177e4SLinus Torvalds /* Security filter */
1313ad254f7SMarcel Holtmann #define HCI_SFLT_MAX_OGF  5
1323ad254f7SMarcel Holtmann 
1333ad254f7SMarcel Holtmann struct hci_sec_filter {
1343ad254f7SMarcel Holtmann 	__u32 type_mask;
1353ad254f7SMarcel Holtmann 	__u32 event_mask[2];
1363ad254f7SMarcel Holtmann 	__u32 ocf_mask[HCI_SFLT_MAX_OGF + 1][4];
1373ad254f7SMarcel Holtmann };
1383ad254f7SMarcel Holtmann 
1397e67c112SMarcel Holtmann static const struct hci_sec_filter hci_sec_filter = {
1401da177e4SLinus Torvalds 	/* Packet types */
1411da177e4SLinus Torvalds 	0x10,
1421da177e4SLinus Torvalds 	/* Events */
143dd7f5527SMarcel Holtmann 	{ 0x1000d9fe, 0x0000b00c },
1441da177e4SLinus Torvalds 	/* Commands */
1451da177e4SLinus Torvalds 	{
1461da177e4SLinus Torvalds 		{ 0x0 },
1471da177e4SLinus Torvalds 		/* OGF_LINK_CTL */
1487c631a67SMarcel Holtmann 		{ 0xbe000006, 0x00000001, 0x00000000, 0x00 },
1491da177e4SLinus Torvalds 		/* OGF_LINK_POLICY */
1507c631a67SMarcel Holtmann 		{ 0x00005200, 0x00000000, 0x00000000, 0x00 },
1511da177e4SLinus Torvalds 		/* OGF_HOST_CTL */
1527c631a67SMarcel Holtmann 		{ 0xaab00200, 0x2b402aaa, 0x05220154, 0x00 },
1531da177e4SLinus Torvalds 		/* OGF_INFO_PARAM */
1547c631a67SMarcel Holtmann 		{ 0x000002be, 0x00000000, 0x00000000, 0x00 },
1551da177e4SLinus Torvalds 		/* OGF_STATUS_PARAM */
1567c631a67SMarcel Holtmann 		{ 0x000000ea, 0x00000000, 0x00000000, 0x00 }
1571da177e4SLinus Torvalds 	}
1581da177e4SLinus Torvalds };
1591da177e4SLinus Torvalds 
1601da177e4SLinus Torvalds static struct bt_sock_list hci_sk_list = {
161d5fb2962SRobert P. J. Day 	.lock = __RW_LOCK_UNLOCKED(hci_sk_list.lock)
1621da177e4SLinus Torvalds };
1631da177e4SLinus Torvalds 
164f81fe64fSMarcel Holtmann static bool is_filtered_packet(struct sock *sk, struct sk_buff *skb)
165f81fe64fSMarcel Holtmann {
166f81fe64fSMarcel Holtmann 	struct hci_filter *flt;
167f81fe64fSMarcel Holtmann 	int flt_type, flt_event;
168f81fe64fSMarcel Holtmann 
169f81fe64fSMarcel Holtmann 	/* Apply filter */
170f81fe64fSMarcel Holtmann 	flt = &hci_pi(sk)->filter;
171f81fe64fSMarcel Holtmann 
172d79f34e3SMarcel Holtmann 	flt_type = hci_skb_pkt_type(skb) & HCI_FLT_TYPE_BITS;
173f81fe64fSMarcel Holtmann 
174f81fe64fSMarcel Holtmann 	if (!test_bit(flt_type, &flt->type_mask))
175f81fe64fSMarcel Holtmann 		return true;
176f81fe64fSMarcel Holtmann 
177f81fe64fSMarcel Holtmann 	/* Extra filter for event packets only */
178d79f34e3SMarcel Holtmann 	if (hci_skb_pkt_type(skb) != HCI_EVENT_PKT)
179f81fe64fSMarcel Holtmann 		return false;
180f81fe64fSMarcel Holtmann 
181f81fe64fSMarcel Holtmann 	flt_event = (*(__u8 *)skb->data & HCI_FLT_EVENT_BITS);
182f81fe64fSMarcel Holtmann 
183f81fe64fSMarcel Holtmann 	if (!hci_test_bit(flt_event, &flt->event_mask))
184f81fe64fSMarcel Holtmann 		return true;
185f81fe64fSMarcel Holtmann 
186f81fe64fSMarcel Holtmann 	/* Check filter only when opcode is set */
187f81fe64fSMarcel Holtmann 	if (!flt->opcode)
188f81fe64fSMarcel Holtmann 		return false;
189f81fe64fSMarcel Holtmann 
190f81fe64fSMarcel Holtmann 	if (flt_event == HCI_EV_CMD_COMPLETE &&
191f81fe64fSMarcel Holtmann 	    flt->opcode != get_unaligned((__le16 *)(skb->data + 3)))
192f81fe64fSMarcel Holtmann 		return true;
193f81fe64fSMarcel Holtmann 
194f81fe64fSMarcel Holtmann 	if (flt_event == HCI_EV_CMD_STATUS &&
195f81fe64fSMarcel Holtmann 	    flt->opcode != get_unaligned((__le16 *)(skb->data + 4)))
196f81fe64fSMarcel Holtmann 		return true;
197f81fe64fSMarcel Holtmann 
198f81fe64fSMarcel Holtmann 	return false;
199f81fe64fSMarcel Holtmann }
200f81fe64fSMarcel Holtmann 
2011da177e4SLinus Torvalds /* Send frame to RAW socket */
202470fe1b5SMarcel Holtmann void hci_send_to_sock(struct hci_dev *hdev, struct sk_buff *skb)
2031da177e4SLinus Torvalds {
2041da177e4SLinus Torvalds 	struct sock *sk;
205e0edf373SMarcel Holtmann 	struct sk_buff *skb_copy = NULL;
2061da177e4SLinus Torvalds 
2071da177e4SLinus Torvalds 	BT_DBG("hdev %p len %d", hdev, skb->len);
2081da177e4SLinus Torvalds 
2091da177e4SLinus Torvalds 	read_lock(&hci_sk_list.lock);
210470fe1b5SMarcel Holtmann 
211b67bfe0dSSasha Levin 	sk_for_each(sk, &hci_sk_list.head) {
2121da177e4SLinus Torvalds 		struct sk_buff *nskb;
2131da177e4SLinus Torvalds 
2141da177e4SLinus Torvalds 		if (sk->sk_state != BT_BOUND || hci_pi(sk)->hdev != hdev)
2151da177e4SLinus Torvalds 			continue;
2161da177e4SLinus Torvalds 
2171da177e4SLinus Torvalds 		/* Don't send frame to the socket it came from */
2181da177e4SLinus Torvalds 		if (skb->sk == sk)
2191da177e4SLinus Torvalds 			continue;
2201da177e4SLinus Torvalds 
22123500189SMarcel Holtmann 		if (hci_pi(sk)->channel == HCI_CHANNEL_RAW) {
222d79f34e3SMarcel Holtmann 			if (hci_skb_pkt_type(skb) != HCI_COMMAND_PKT &&
223d79f34e3SMarcel Holtmann 			    hci_skb_pkt_type(skb) != HCI_EVENT_PKT &&
224d79f34e3SMarcel Holtmann 			    hci_skb_pkt_type(skb) != HCI_ACLDATA_PKT &&
225cc974003SMarcel Holtmann 			    hci_skb_pkt_type(skb) != HCI_SCODATA_PKT &&
226cc974003SMarcel Holtmann 			    hci_skb_pkt_type(skb) != HCI_ISODATA_PKT)
227bb77543eSMarcel Holtmann 				continue;
228f81fe64fSMarcel Holtmann 			if (is_filtered_packet(sk, skb))
2291da177e4SLinus Torvalds 				continue;
23023500189SMarcel Holtmann 		} else if (hci_pi(sk)->channel == HCI_CHANNEL_USER) {
23123500189SMarcel Holtmann 			if (!bt_cb(skb)->incoming)
23223500189SMarcel Holtmann 				continue;
233d79f34e3SMarcel Holtmann 			if (hci_skb_pkt_type(skb) != HCI_EVENT_PKT &&
234d79f34e3SMarcel Holtmann 			    hci_skb_pkt_type(skb) != HCI_ACLDATA_PKT &&
235cc974003SMarcel Holtmann 			    hci_skb_pkt_type(skb) != HCI_SCODATA_PKT &&
236cc974003SMarcel Holtmann 			    hci_skb_pkt_type(skb) != HCI_ISODATA_PKT)
23723500189SMarcel Holtmann 				continue;
23823500189SMarcel Holtmann 		} else {
23923500189SMarcel Holtmann 			/* Don't send frame to other channel types */
24023500189SMarcel Holtmann 			continue;
24123500189SMarcel Holtmann 		}
2421da177e4SLinus Torvalds 
243e0edf373SMarcel Holtmann 		if (!skb_copy) {
244e0edf373SMarcel Holtmann 			/* Create a private copy with headroom */
245bad93e9dSOctavian Purdila 			skb_copy = __pskb_copy_fclone(skb, 1, GFP_ATOMIC, true);
246e0edf373SMarcel Holtmann 			if (!skb_copy)
2471da177e4SLinus Torvalds 				continue;
2481da177e4SLinus Torvalds 
2491da177e4SLinus Torvalds 			/* Put type byte before the data */
250d79f34e3SMarcel Holtmann 			memcpy(skb_push(skb_copy, 1), &hci_skb_pkt_type(skb), 1);
251e0edf373SMarcel Holtmann 		}
252e0edf373SMarcel Holtmann 
253e0edf373SMarcel Holtmann 		nskb = skb_clone(skb_copy, GFP_ATOMIC);
254e0edf373SMarcel Holtmann 		if (!nskb)
255e0edf373SMarcel Holtmann 			continue;
2561da177e4SLinus Torvalds 
2571da177e4SLinus Torvalds 		if (sock_queue_rcv_skb(sk, nskb))
2581da177e4SLinus Torvalds 			kfree_skb(nskb);
2591da177e4SLinus Torvalds 	}
260470fe1b5SMarcel Holtmann 
261470fe1b5SMarcel Holtmann 	read_unlock(&hci_sk_list.lock);
262e0edf373SMarcel Holtmann 
263e0edf373SMarcel Holtmann 	kfree_skb(skb_copy);
264470fe1b5SMarcel Holtmann }
265470fe1b5SMarcel Holtmann 
2667129069eSJohan Hedberg /* Send frame to sockets with specific channel */
267a9ee77afSSebastian Andrzej Siewior static void __hci_send_to_channel(unsigned short channel, struct sk_buff *skb,
268c08b1a1dSMarcel Holtmann 				  int flag, struct sock *skip_sk)
269470fe1b5SMarcel Holtmann {
270470fe1b5SMarcel Holtmann 	struct sock *sk;
271470fe1b5SMarcel Holtmann 
2727129069eSJohan Hedberg 	BT_DBG("channel %u len %d", channel, skb->len);
273470fe1b5SMarcel Holtmann 
274b67bfe0dSSasha Levin 	sk_for_each(sk, &hci_sk_list.head) {
275470fe1b5SMarcel Holtmann 		struct sk_buff *nskb;
276470fe1b5SMarcel Holtmann 
277c08b1a1dSMarcel Holtmann 		/* Ignore socket without the flag set */
278c85be545SMarcel Holtmann 		if (!hci_sock_test_flag(sk, flag))
279c08b1a1dSMarcel Holtmann 			continue;
280c08b1a1dSMarcel Holtmann 
281470fe1b5SMarcel Holtmann 		/* Skip the original socket */
282470fe1b5SMarcel Holtmann 		if (sk == skip_sk)
283470fe1b5SMarcel Holtmann 			continue;
284470fe1b5SMarcel Holtmann 
285470fe1b5SMarcel Holtmann 		if (sk->sk_state != BT_BOUND)
286470fe1b5SMarcel Holtmann 			continue;
287470fe1b5SMarcel Holtmann 
2887129069eSJohan Hedberg 		if (hci_pi(sk)->channel != channel)
289d7f72f61SMarcel Holtmann 			continue;
290d7f72f61SMarcel Holtmann 
291d7f72f61SMarcel Holtmann 		nskb = skb_clone(skb, GFP_ATOMIC);
292d7f72f61SMarcel Holtmann 		if (!nskb)
293d7f72f61SMarcel Holtmann 			continue;
294d7f72f61SMarcel Holtmann 
295d7f72f61SMarcel Holtmann 		if (sock_queue_rcv_skb(sk, nskb))
296d7f72f61SMarcel Holtmann 			kfree_skb(nskb);
297d7f72f61SMarcel Holtmann 	}
298d7f72f61SMarcel Holtmann 
299a9ee77afSSebastian Andrzej Siewior }
300a9ee77afSSebastian Andrzej Siewior 
301a9ee77afSSebastian Andrzej Siewior void hci_send_to_channel(unsigned short channel, struct sk_buff *skb,
302a9ee77afSSebastian Andrzej Siewior 			 int flag, struct sock *skip_sk)
303a9ee77afSSebastian Andrzej Siewior {
304a9ee77afSSebastian Andrzej Siewior 	read_lock(&hci_sk_list.lock);
305a9ee77afSSebastian Andrzej Siewior 	__hci_send_to_channel(channel, skb, flag, skip_sk);
306d7f72f61SMarcel Holtmann 	read_unlock(&hci_sk_list.lock);
307d7f72f61SMarcel Holtmann }
308d7f72f61SMarcel Holtmann 
309cd82e61cSMarcel Holtmann /* Send frame to monitor socket */
310cd82e61cSMarcel Holtmann void hci_send_to_monitor(struct hci_dev *hdev, struct sk_buff *skb)
311cd82e61cSMarcel Holtmann {
312cd82e61cSMarcel Holtmann 	struct sk_buff *skb_copy = NULL;
3132b531294SMarcel Holtmann 	struct hci_mon_hdr *hdr;
314cd82e61cSMarcel Holtmann 	__le16 opcode;
315cd82e61cSMarcel Holtmann 
316cd82e61cSMarcel Holtmann 	if (!atomic_read(&monitor_promisc))
317cd82e61cSMarcel Holtmann 		return;
318cd82e61cSMarcel Holtmann 
319cd82e61cSMarcel Holtmann 	BT_DBG("hdev %p len %d", hdev, skb->len);
320cd82e61cSMarcel Holtmann 
321d79f34e3SMarcel Holtmann 	switch (hci_skb_pkt_type(skb)) {
322cd82e61cSMarcel Holtmann 	case HCI_COMMAND_PKT:
323dcf4adbfSJoe Perches 		opcode = cpu_to_le16(HCI_MON_COMMAND_PKT);
324cd82e61cSMarcel Holtmann 		break;
325cd82e61cSMarcel Holtmann 	case HCI_EVENT_PKT:
326dcf4adbfSJoe Perches 		opcode = cpu_to_le16(HCI_MON_EVENT_PKT);
327cd82e61cSMarcel Holtmann 		break;
328cd82e61cSMarcel Holtmann 	case HCI_ACLDATA_PKT:
329cd82e61cSMarcel Holtmann 		if (bt_cb(skb)->incoming)
330dcf4adbfSJoe Perches 			opcode = cpu_to_le16(HCI_MON_ACL_RX_PKT);
331cd82e61cSMarcel Holtmann 		else
332dcf4adbfSJoe Perches 			opcode = cpu_to_le16(HCI_MON_ACL_TX_PKT);
333cd82e61cSMarcel Holtmann 		break;
334cd82e61cSMarcel Holtmann 	case HCI_SCODATA_PKT:
335cd82e61cSMarcel Holtmann 		if (bt_cb(skb)->incoming)
336dcf4adbfSJoe Perches 			opcode = cpu_to_le16(HCI_MON_SCO_RX_PKT);
337cd82e61cSMarcel Holtmann 		else
338dcf4adbfSJoe Perches 			opcode = cpu_to_le16(HCI_MON_SCO_TX_PKT);
339cd82e61cSMarcel Holtmann 		break;
340f9a619dbSLuiz Augusto von Dentz 	case HCI_ISODATA_PKT:
341f9a619dbSLuiz Augusto von Dentz 		if (bt_cb(skb)->incoming)
342f9a619dbSLuiz Augusto von Dentz 			opcode = cpu_to_le16(HCI_MON_ISO_RX_PKT);
343f9a619dbSLuiz Augusto von Dentz 		else
344f9a619dbSLuiz Augusto von Dentz 			opcode = cpu_to_le16(HCI_MON_ISO_TX_PKT);
345f9a619dbSLuiz Augusto von Dentz 		break;
346e875ff84SMarcel Holtmann 	case HCI_DIAG_PKT:
347e875ff84SMarcel Holtmann 		opcode = cpu_to_le16(HCI_MON_VENDOR_DIAG);
348e875ff84SMarcel Holtmann 		break;
349cd82e61cSMarcel Holtmann 	default:
350cd82e61cSMarcel Holtmann 		return;
351cd82e61cSMarcel Holtmann 	}
352cd82e61cSMarcel Holtmann 
3532b531294SMarcel Holtmann 	/* Create a private copy with headroom */
3542b531294SMarcel Holtmann 	skb_copy = __pskb_copy_fclone(skb, HCI_MON_HDR_SIZE, GFP_ATOMIC, true);
3552b531294SMarcel Holtmann 	if (!skb_copy)
3562b531294SMarcel Holtmann 		return;
3572b531294SMarcel Holtmann 
3582b531294SMarcel Holtmann 	/* Put header before the data */
359d58ff351SJohannes Berg 	hdr = skb_push(skb_copy, HCI_MON_HDR_SIZE);
3602b531294SMarcel Holtmann 	hdr->opcode = opcode;
3612b531294SMarcel Holtmann 	hdr->index = cpu_to_le16(hdev->id);
3622b531294SMarcel Holtmann 	hdr->len = cpu_to_le16(skb->len);
3632b531294SMarcel Holtmann 
364c08b1a1dSMarcel Holtmann 	hci_send_to_channel(HCI_CHANNEL_MONITOR, skb_copy,
365c08b1a1dSMarcel Holtmann 			    HCI_SOCK_TRUSTED, NULL);
366cd82e61cSMarcel Holtmann 	kfree_skb(skb_copy);
367cd82e61cSMarcel Holtmann }
368cd82e61cSMarcel Holtmann 
36938ceaa00SMarcel Holtmann void hci_send_monitor_ctrl_event(struct hci_dev *hdev, u16 event,
37038ceaa00SMarcel Holtmann 				 void *data, u16 data_len, ktime_t tstamp,
37138ceaa00SMarcel Holtmann 				 int flag, struct sock *skip_sk)
37238ceaa00SMarcel Holtmann {
37338ceaa00SMarcel Holtmann 	struct sock *sk;
37438ceaa00SMarcel Holtmann 	__le16 index;
37538ceaa00SMarcel Holtmann 
37638ceaa00SMarcel Holtmann 	if (hdev)
37738ceaa00SMarcel Holtmann 		index = cpu_to_le16(hdev->id);
37838ceaa00SMarcel Holtmann 	else
37938ceaa00SMarcel Holtmann 		index = cpu_to_le16(MGMT_INDEX_NONE);
38038ceaa00SMarcel Holtmann 
38138ceaa00SMarcel Holtmann 	read_lock(&hci_sk_list.lock);
38238ceaa00SMarcel Holtmann 
38338ceaa00SMarcel Holtmann 	sk_for_each(sk, &hci_sk_list.head) {
38438ceaa00SMarcel Holtmann 		struct hci_mon_hdr *hdr;
38538ceaa00SMarcel Holtmann 		struct sk_buff *skb;
38638ceaa00SMarcel Holtmann 
38738ceaa00SMarcel Holtmann 		if (hci_pi(sk)->channel != HCI_CHANNEL_CONTROL)
38838ceaa00SMarcel Holtmann 			continue;
38938ceaa00SMarcel Holtmann 
39038ceaa00SMarcel Holtmann 		/* Ignore socket without the flag set */
39138ceaa00SMarcel Holtmann 		if (!hci_sock_test_flag(sk, flag))
39238ceaa00SMarcel Holtmann 			continue;
39338ceaa00SMarcel Holtmann 
39438ceaa00SMarcel Holtmann 		/* Skip the original socket */
39538ceaa00SMarcel Holtmann 		if (sk == skip_sk)
39638ceaa00SMarcel Holtmann 			continue;
39738ceaa00SMarcel Holtmann 
39838ceaa00SMarcel Holtmann 		skb = bt_skb_alloc(6 + data_len, GFP_ATOMIC);
39938ceaa00SMarcel Holtmann 		if (!skb)
40038ceaa00SMarcel Holtmann 			continue;
40138ceaa00SMarcel Holtmann 
40238ceaa00SMarcel Holtmann 		put_unaligned_le32(hci_pi(sk)->cookie, skb_put(skb, 4));
40338ceaa00SMarcel Holtmann 		put_unaligned_le16(event, skb_put(skb, 2));
40438ceaa00SMarcel Holtmann 
40538ceaa00SMarcel Holtmann 		if (data)
40659ae1d12SJohannes Berg 			skb_put_data(skb, data, data_len);
40738ceaa00SMarcel Holtmann 
40838ceaa00SMarcel Holtmann 		skb->tstamp = tstamp;
40938ceaa00SMarcel Holtmann 
410d58ff351SJohannes Berg 		hdr = skb_push(skb, HCI_MON_HDR_SIZE);
41138ceaa00SMarcel Holtmann 		hdr->opcode = cpu_to_le16(HCI_MON_CTRL_EVENT);
41238ceaa00SMarcel Holtmann 		hdr->index = index;
41338ceaa00SMarcel Holtmann 		hdr->len = cpu_to_le16(skb->len - HCI_MON_HDR_SIZE);
41438ceaa00SMarcel Holtmann 
415a9ee77afSSebastian Andrzej Siewior 		__hci_send_to_channel(HCI_CHANNEL_MONITOR, skb,
41638ceaa00SMarcel Holtmann 				      HCI_SOCK_TRUSTED, NULL);
41738ceaa00SMarcel Holtmann 		kfree_skb(skb);
41838ceaa00SMarcel Holtmann 	}
41938ceaa00SMarcel Holtmann 
42038ceaa00SMarcel Holtmann 	read_unlock(&hci_sk_list.lock);
42138ceaa00SMarcel Holtmann }
42238ceaa00SMarcel Holtmann 
423cd82e61cSMarcel Holtmann static struct sk_buff *create_monitor_event(struct hci_dev *hdev, int event)
424cd82e61cSMarcel Holtmann {
425cd82e61cSMarcel Holtmann 	struct hci_mon_hdr *hdr;
426cd82e61cSMarcel Holtmann 	struct hci_mon_new_index *ni;
4276c566dd5SMarcel Holtmann 	struct hci_mon_index_info *ii;
428cd82e61cSMarcel Holtmann 	struct sk_buff *skb;
429cd82e61cSMarcel Holtmann 	__le16 opcode;
430cd82e61cSMarcel Holtmann 
431cd82e61cSMarcel Holtmann 	switch (event) {
432cd82e61cSMarcel Holtmann 	case HCI_DEV_REG:
433cd82e61cSMarcel Holtmann 		skb = bt_skb_alloc(HCI_MON_NEW_INDEX_SIZE, GFP_ATOMIC);
434cd82e61cSMarcel Holtmann 		if (!skb)
435cd82e61cSMarcel Holtmann 			return NULL;
436cd82e61cSMarcel Holtmann 
4374df864c1SJohannes Berg 		ni = skb_put(skb, HCI_MON_NEW_INDEX_SIZE);
438cd82e61cSMarcel Holtmann 		ni->type = hdev->dev_type;
439cd82e61cSMarcel Holtmann 		ni->bus = hdev->bus;
440cd82e61cSMarcel Holtmann 		bacpy(&ni->bdaddr, &hdev->bdaddr);
441cd82e61cSMarcel Holtmann 		memcpy(ni->name, hdev->name, 8);
442cd82e61cSMarcel Holtmann 
443dcf4adbfSJoe Perches 		opcode = cpu_to_le16(HCI_MON_NEW_INDEX);
444cd82e61cSMarcel Holtmann 		break;
445cd82e61cSMarcel Holtmann 
446cd82e61cSMarcel Holtmann 	case HCI_DEV_UNREG:
447cd82e61cSMarcel Holtmann 		skb = bt_skb_alloc(0, GFP_ATOMIC);
448cd82e61cSMarcel Holtmann 		if (!skb)
449cd82e61cSMarcel Holtmann 			return NULL;
450cd82e61cSMarcel Holtmann 
451dcf4adbfSJoe Perches 		opcode = cpu_to_le16(HCI_MON_DEL_INDEX);
452cd82e61cSMarcel Holtmann 		break;
453cd82e61cSMarcel Holtmann 
454e131d74aSMarcel Holtmann 	case HCI_DEV_SETUP:
455e131d74aSMarcel Holtmann 		if (hdev->manufacturer == 0xffff)
456e131d74aSMarcel Holtmann 			return NULL;
45719186c7bSGustavo A. R. Silva 		fallthrough;
458e131d74aSMarcel Holtmann 
4596c566dd5SMarcel Holtmann 	case HCI_DEV_UP:
4606c566dd5SMarcel Holtmann 		skb = bt_skb_alloc(HCI_MON_INDEX_INFO_SIZE, GFP_ATOMIC);
4616c566dd5SMarcel Holtmann 		if (!skb)
4626c566dd5SMarcel Holtmann 			return NULL;
4636c566dd5SMarcel Holtmann 
4644df864c1SJohannes Berg 		ii = skb_put(skb, HCI_MON_INDEX_INFO_SIZE);
4656c566dd5SMarcel Holtmann 		bacpy(&ii->bdaddr, &hdev->bdaddr);
4666c566dd5SMarcel Holtmann 		ii->manufacturer = cpu_to_le16(hdev->manufacturer);
4676c566dd5SMarcel Holtmann 
4686c566dd5SMarcel Holtmann 		opcode = cpu_to_le16(HCI_MON_INDEX_INFO);
4696c566dd5SMarcel Holtmann 		break;
4706c566dd5SMarcel Holtmann 
47122db3cbcSMarcel Holtmann 	case HCI_DEV_OPEN:
47222db3cbcSMarcel Holtmann 		skb = bt_skb_alloc(0, GFP_ATOMIC);
47322db3cbcSMarcel Holtmann 		if (!skb)
47422db3cbcSMarcel Holtmann 			return NULL;
47522db3cbcSMarcel Holtmann 
47622db3cbcSMarcel Holtmann 		opcode = cpu_to_le16(HCI_MON_OPEN_INDEX);
47722db3cbcSMarcel Holtmann 		break;
47822db3cbcSMarcel Holtmann 
47922db3cbcSMarcel Holtmann 	case HCI_DEV_CLOSE:
48022db3cbcSMarcel Holtmann 		skb = bt_skb_alloc(0, GFP_ATOMIC);
48122db3cbcSMarcel Holtmann 		if (!skb)
48222db3cbcSMarcel Holtmann 			return NULL;
48322db3cbcSMarcel Holtmann 
48422db3cbcSMarcel Holtmann 		opcode = cpu_to_le16(HCI_MON_CLOSE_INDEX);
48522db3cbcSMarcel Holtmann 		break;
48622db3cbcSMarcel Holtmann 
487cd82e61cSMarcel Holtmann 	default:
488cd82e61cSMarcel Holtmann 		return NULL;
489cd82e61cSMarcel Holtmann 	}
490cd82e61cSMarcel Holtmann 
491cd82e61cSMarcel Holtmann 	__net_timestamp(skb);
492cd82e61cSMarcel Holtmann 
493d58ff351SJohannes Berg 	hdr = skb_push(skb, HCI_MON_HDR_SIZE);
494cd82e61cSMarcel Holtmann 	hdr->opcode = opcode;
495cd82e61cSMarcel Holtmann 	hdr->index = cpu_to_le16(hdev->id);
496cd82e61cSMarcel Holtmann 	hdr->len = cpu_to_le16(skb->len - HCI_MON_HDR_SIZE);
497cd82e61cSMarcel Holtmann 
498cd82e61cSMarcel Holtmann 	return skb;
499cd82e61cSMarcel Holtmann }
500cd82e61cSMarcel Holtmann 
501249fa169SMarcel Holtmann static struct sk_buff *create_monitor_ctrl_open(struct sock *sk)
502249fa169SMarcel Holtmann {
503249fa169SMarcel Holtmann 	struct hci_mon_hdr *hdr;
504249fa169SMarcel Holtmann 	struct sk_buff *skb;
505d0bef1d2SMarcel Holtmann 	u16 format;
506249fa169SMarcel Holtmann 	u8 ver[3];
507249fa169SMarcel Holtmann 	u32 flags;
508249fa169SMarcel Holtmann 
5090ef2c42fSMarcel Holtmann 	/* No message needed when cookie is not present */
5100ef2c42fSMarcel Holtmann 	if (!hci_pi(sk)->cookie)
5110ef2c42fSMarcel Holtmann 		return NULL;
5120ef2c42fSMarcel Holtmann 
513d0bef1d2SMarcel Holtmann 	switch (hci_pi(sk)->channel) {
514f81f5b2dSMarcel Holtmann 	case HCI_CHANNEL_RAW:
515f81f5b2dSMarcel Holtmann 		format = 0x0000;
516f81f5b2dSMarcel Holtmann 		ver[0] = BT_SUBSYS_VERSION;
517f81f5b2dSMarcel Holtmann 		put_unaligned_le16(BT_SUBSYS_REVISION, ver + 1);
518f81f5b2dSMarcel Holtmann 		break;
519aa1638ddSMarcel Holtmann 	case HCI_CHANNEL_USER:
520aa1638ddSMarcel Holtmann 		format = 0x0001;
521aa1638ddSMarcel Holtmann 		ver[0] = BT_SUBSYS_VERSION;
522aa1638ddSMarcel Holtmann 		put_unaligned_le16(BT_SUBSYS_REVISION, ver + 1);
523aa1638ddSMarcel Holtmann 		break;
524d0bef1d2SMarcel Holtmann 	case HCI_CHANNEL_CONTROL:
525d0bef1d2SMarcel Holtmann 		format = 0x0002;
526d0bef1d2SMarcel Holtmann 		mgmt_fill_version_info(ver);
527d0bef1d2SMarcel Holtmann 		break;
528d0bef1d2SMarcel Holtmann 	default:
529d0bef1d2SMarcel Holtmann 		/* No message for unsupported format */
530d0bef1d2SMarcel Holtmann 		return NULL;
531d0bef1d2SMarcel Holtmann 	}
532d0bef1d2SMarcel Holtmann 
533249fa169SMarcel Holtmann 	skb = bt_skb_alloc(14 + TASK_COMM_LEN , GFP_ATOMIC);
534249fa169SMarcel Holtmann 	if (!skb)
535249fa169SMarcel Holtmann 		return NULL;
536249fa169SMarcel Holtmann 
537249fa169SMarcel Holtmann 	flags = hci_sock_test_flag(sk, HCI_SOCK_TRUSTED) ? 0x1 : 0x0;
538249fa169SMarcel Holtmann 
539249fa169SMarcel Holtmann 	put_unaligned_le32(hci_pi(sk)->cookie, skb_put(skb, 4));
540249fa169SMarcel Holtmann 	put_unaligned_le16(format, skb_put(skb, 2));
54159ae1d12SJohannes Berg 	skb_put_data(skb, ver, sizeof(ver));
542249fa169SMarcel Holtmann 	put_unaligned_le32(flags, skb_put(skb, 4));
543634fef61SJohannes Berg 	skb_put_u8(skb, TASK_COMM_LEN);
54459ae1d12SJohannes Berg 	skb_put_data(skb, hci_pi(sk)->comm, TASK_COMM_LEN);
545249fa169SMarcel Holtmann 
546249fa169SMarcel Holtmann 	__net_timestamp(skb);
547249fa169SMarcel Holtmann 
548d58ff351SJohannes Berg 	hdr = skb_push(skb, HCI_MON_HDR_SIZE);
549249fa169SMarcel Holtmann 	hdr->opcode = cpu_to_le16(HCI_MON_CTRL_OPEN);
5500ef2c42fSMarcel Holtmann 	if (hci_pi(sk)->hdev)
5510ef2c42fSMarcel Holtmann 		hdr->index = cpu_to_le16(hci_pi(sk)->hdev->id);
5520ef2c42fSMarcel Holtmann 	else
553249fa169SMarcel Holtmann 		hdr->index = cpu_to_le16(HCI_DEV_NONE);
554249fa169SMarcel Holtmann 	hdr->len = cpu_to_le16(skb->len - HCI_MON_HDR_SIZE);
555249fa169SMarcel Holtmann 
556249fa169SMarcel Holtmann 	return skb;
557249fa169SMarcel Holtmann }
558249fa169SMarcel Holtmann 
559249fa169SMarcel Holtmann static struct sk_buff *create_monitor_ctrl_close(struct sock *sk)
560249fa169SMarcel Holtmann {
561249fa169SMarcel Holtmann 	struct hci_mon_hdr *hdr;
562249fa169SMarcel Holtmann 	struct sk_buff *skb;
563249fa169SMarcel Holtmann 
5640ef2c42fSMarcel Holtmann 	/* No message needed when cookie is not present */
5650ef2c42fSMarcel Holtmann 	if (!hci_pi(sk)->cookie)
5660ef2c42fSMarcel Holtmann 		return NULL;
5670ef2c42fSMarcel Holtmann 
568d0bef1d2SMarcel Holtmann 	switch (hci_pi(sk)->channel) {
569f81f5b2dSMarcel Holtmann 	case HCI_CHANNEL_RAW:
570aa1638ddSMarcel Holtmann 	case HCI_CHANNEL_USER:
571d0bef1d2SMarcel Holtmann 	case HCI_CHANNEL_CONTROL:
572d0bef1d2SMarcel Holtmann 		break;
573d0bef1d2SMarcel Holtmann 	default:
574d0bef1d2SMarcel Holtmann 		/* No message for unsupported format */
575d0bef1d2SMarcel Holtmann 		return NULL;
576d0bef1d2SMarcel Holtmann 	}
577d0bef1d2SMarcel Holtmann 
578249fa169SMarcel Holtmann 	skb = bt_skb_alloc(4, GFP_ATOMIC);
579249fa169SMarcel Holtmann 	if (!skb)
580249fa169SMarcel Holtmann 		return NULL;
581249fa169SMarcel Holtmann 
582249fa169SMarcel Holtmann 	put_unaligned_le32(hci_pi(sk)->cookie, skb_put(skb, 4));
583249fa169SMarcel Holtmann 
584249fa169SMarcel Holtmann 	__net_timestamp(skb);
585249fa169SMarcel Holtmann 
586d58ff351SJohannes Berg 	hdr = skb_push(skb, HCI_MON_HDR_SIZE);
587249fa169SMarcel Holtmann 	hdr->opcode = cpu_to_le16(HCI_MON_CTRL_CLOSE);
5880ef2c42fSMarcel Holtmann 	if (hci_pi(sk)->hdev)
5890ef2c42fSMarcel Holtmann 		hdr->index = cpu_to_le16(hci_pi(sk)->hdev->id);
5900ef2c42fSMarcel Holtmann 	else
591249fa169SMarcel Holtmann 		hdr->index = cpu_to_le16(HCI_DEV_NONE);
592249fa169SMarcel Holtmann 	hdr->len = cpu_to_le16(skb->len - HCI_MON_HDR_SIZE);
593249fa169SMarcel Holtmann 
594249fa169SMarcel Holtmann 	return skb;
595249fa169SMarcel Holtmann }
596249fa169SMarcel Holtmann 
59738ceaa00SMarcel Holtmann static struct sk_buff *create_monitor_ctrl_command(struct sock *sk, u16 index,
59838ceaa00SMarcel Holtmann 						   u16 opcode, u16 len,
59938ceaa00SMarcel Holtmann 						   const void *buf)
60038ceaa00SMarcel Holtmann {
60138ceaa00SMarcel Holtmann 	struct hci_mon_hdr *hdr;
60238ceaa00SMarcel Holtmann 	struct sk_buff *skb;
60338ceaa00SMarcel Holtmann 
60438ceaa00SMarcel Holtmann 	skb = bt_skb_alloc(6 + len, GFP_ATOMIC);
60538ceaa00SMarcel Holtmann 	if (!skb)
60638ceaa00SMarcel Holtmann 		return NULL;
60738ceaa00SMarcel Holtmann 
60838ceaa00SMarcel Holtmann 	put_unaligned_le32(hci_pi(sk)->cookie, skb_put(skb, 4));
60938ceaa00SMarcel Holtmann 	put_unaligned_le16(opcode, skb_put(skb, 2));
61038ceaa00SMarcel Holtmann 
61138ceaa00SMarcel Holtmann 	if (buf)
61259ae1d12SJohannes Berg 		skb_put_data(skb, buf, len);
61338ceaa00SMarcel Holtmann 
61438ceaa00SMarcel Holtmann 	__net_timestamp(skb);
61538ceaa00SMarcel Holtmann 
616d58ff351SJohannes Berg 	hdr = skb_push(skb, HCI_MON_HDR_SIZE);
61738ceaa00SMarcel Holtmann 	hdr->opcode = cpu_to_le16(HCI_MON_CTRL_COMMAND);
61838ceaa00SMarcel Holtmann 	hdr->index = cpu_to_le16(index);
61938ceaa00SMarcel Holtmann 	hdr->len = cpu_to_le16(skb->len - HCI_MON_HDR_SIZE);
62038ceaa00SMarcel Holtmann 
62138ceaa00SMarcel Holtmann 	return skb;
62238ceaa00SMarcel Holtmann }
62338ceaa00SMarcel Holtmann 
624787b306cSJohannes Berg static void __printf(2, 3)
625787b306cSJohannes Berg send_monitor_note(struct sock *sk, const char *fmt, ...)
626dd31506dSMarcel Holtmann {
627787b306cSJohannes Berg 	size_t len;
628dd31506dSMarcel Holtmann 	struct hci_mon_hdr *hdr;
629dd31506dSMarcel Holtmann 	struct sk_buff *skb;
630787b306cSJohannes Berg 	va_list args;
631787b306cSJohannes Berg 
632787b306cSJohannes Berg 	va_start(args, fmt);
633787b306cSJohannes Berg 	len = vsnprintf(NULL, 0, fmt, args);
634787b306cSJohannes Berg 	va_end(args);
635dd31506dSMarcel Holtmann 
636dd31506dSMarcel Holtmann 	skb = bt_skb_alloc(len + 1, GFP_ATOMIC);
637dd31506dSMarcel Holtmann 	if (!skb)
638dd31506dSMarcel Holtmann 		return;
639dd31506dSMarcel Holtmann 
640787b306cSJohannes Berg 	va_start(args, fmt);
641787b306cSJohannes Berg 	vsprintf(skb_put(skb, len), fmt, args);
6424df864c1SJohannes Berg 	*(u8 *)skb_put(skb, 1) = 0;
643787b306cSJohannes Berg 	va_end(args);
644dd31506dSMarcel Holtmann 
645dd31506dSMarcel Holtmann 	__net_timestamp(skb);
646dd31506dSMarcel Holtmann 
647dd31506dSMarcel Holtmann 	hdr = (void *)skb_push(skb, HCI_MON_HDR_SIZE);
648dd31506dSMarcel Holtmann 	hdr->opcode = cpu_to_le16(HCI_MON_SYSTEM_NOTE);
649dd31506dSMarcel Holtmann 	hdr->index = cpu_to_le16(HCI_DEV_NONE);
650dd31506dSMarcel Holtmann 	hdr->len = cpu_to_le16(skb->len - HCI_MON_HDR_SIZE);
651dd31506dSMarcel Holtmann 
652dd31506dSMarcel Holtmann 	if (sock_queue_rcv_skb(sk, skb))
653dd31506dSMarcel Holtmann 		kfree_skb(skb);
654dd31506dSMarcel Holtmann }
655dd31506dSMarcel Holtmann 
656cd82e61cSMarcel Holtmann static void send_monitor_replay(struct sock *sk)
657cd82e61cSMarcel Holtmann {
658cd82e61cSMarcel Holtmann 	struct hci_dev *hdev;
659cd82e61cSMarcel Holtmann 
660cd82e61cSMarcel Holtmann 	read_lock(&hci_dev_list_lock);
661cd82e61cSMarcel Holtmann 
662cd82e61cSMarcel Holtmann 	list_for_each_entry(hdev, &hci_dev_list, list) {
663cd82e61cSMarcel Holtmann 		struct sk_buff *skb;
664cd82e61cSMarcel Holtmann 
665cd82e61cSMarcel Holtmann 		skb = create_monitor_event(hdev, HCI_DEV_REG);
666cd82e61cSMarcel Holtmann 		if (!skb)
667cd82e61cSMarcel Holtmann 			continue;
668cd82e61cSMarcel Holtmann 
669cd82e61cSMarcel Holtmann 		if (sock_queue_rcv_skb(sk, skb))
670cd82e61cSMarcel Holtmann 			kfree_skb(skb);
67122db3cbcSMarcel Holtmann 
67222db3cbcSMarcel Holtmann 		if (!test_bit(HCI_RUNNING, &hdev->flags))
67322db3cbcSMarcel Holtmann 			continue;
67422db3cbcSMarcel Holtmann 
67522db3cbcSMarcel Holtmann 		skb = create_monitor_event(hdev, HCI_DEV_OPEN);
67622db3cbcSMarcel Holtmann 		if (!skb)
67722db3cbcSMarcel Holtmann 			continue;
67822db3cbcSMarcel Holtmann 
67922db3cbcSMarcel Holtmann 		if (sock_queue_rcv_skb(sk, skb))
68022db3cbcSMarcel Holtmann 			kfree_skb(skb);
6816c566dd5SMarcel Holtmann 
682e131d74aSMarcel Holtmann 		if (test_bit(HCI_UP, &hdev->flags))
6836c566dd5SMarcel Holtmann 			skb = create_monitor_event(hdev, HCI_DEV_UP);
684e131d74aSMarcel Holtmann 		else if (hci_dev_test_flag(hdev, HCI_SETUP))
685e131d74aSMarcel Holtmann 			skb = create_monitor_event(hdev, HCI_DEV_SETUP);
686e131d74aSMarcel Holtmann 		else
687e131d74aSMarcel Holtmann 			skb = NULL;
6886c566dd5SMarcel Holtmann 
689e131d74aSMarcel Holtmann 		if (skb) {
6906c566dd5SMarcel Holtmann 			if (sock_queue_rcv_skb(sk, skb))
6916c566dd5SMarcel Holtmann 				kfree_skb(skb);
692cd82e61cSMarcel Holtmann 		}
693e131d74aSMarcel Holtmann 	}
694cd82e61cSMarcel Holtmann 
695cd82e61cSMarcel Holtmann 	read_unlock(&hci_dev_list_lock);
696cd82e61cSMarcel Holtmann }
697cd82e61cSMarcel Holtmann 
698249fa169SMarcel Holtmann static void send_monitor_control_replay(struct sock *mon_sk)
699249fa169SMarcel Holtmann {
700249fa169SMarcel Holtmann 	struct sock *sk;
701249fa169SMarcel Holtmann 
702249fa169SMarcel Holtmann 	read_lock(&hci_sk_list.lock);
703249fa169SMarcel Holtmann 
704249fa169SMarcel Holtmann 	sk_for_each(sk, &hci_sk_list.head) {
705249fa169SMarcel Holtmann 		struct sk_buff *skb;
706249fa169SMarcel Holtmann 
707249fa169SMarcel Holtmann 		skb = create_monitor_ctrl_open(sk);
708249fa169SMarcel Holtmann 		if (!skb)
709249fa169SMarcel Holtmann 			continue;
710249fa169SMarcel Holtmann 
711249fa169SMarcel Holtmann 		if (sock_queue_rcv_skb(mon_sk, skb))
712249fa169SMarcel Holtmann 			kfree_skb(skb);
713249fa169SMarcel Holtmann 	}
714249fa169SMarcel Holtmann 
715249fa169SMarcel Holtmann 	read_unlock(&hci_sk_list.lock);
716249fa169SMarcel Holtmann }
717249fa169SMarcel Holtmann 
718040030efSMarcel Holtmann /* Generate internal stack event */
719040030efSMarcel Holtmann static void hci_si_event(struct hci_dev *hdev, int type, int dlen, void *data)
720040030efSMarcel Holtmann {
721040030efSMarcel Holtmann 	struct hci_event_hdr *hdr;
722040030efSMarcel Holtmann 	struct hci_ev_stack_internal *ev;
723040030efSMarcel Holtmann 	struct sk_buff *skb;
724040030efSMarcel Holtmann 
725040030efSMarcel Holtmann 	skb = bt_skb_alloc(HCI_EVENT_HDR_SIZE + sizeof(*ev) + dlen, GFP_ATOMIC);
726040030efSMarcel Holtmann 	if (!skb)
727040030efSMarcel Holtmann 		return;
728040030efSMarcel Holtmann 
7294df864c1SJohannes Berg 	hdr = skb_put(skb, HCI_EVENT_HDR_SIZE);
730040030efSMarcel Holtmann 	hdr->evt  = HCI_EV_STACK_INTERNAL;
731040030efSMarcel Holtmann 	hdr->plen = sizeof(*ev) + dlen;
732040030efSMarcel Holtmann 
7334df864c1SJohannes Berg 	ev = skb_put(skb, sizeof(*ev) + dlen);
734040030efSMarcel Holtmann 	ev->type = type;
735040030efSMarcel Holtmann 	memcpy(ev->data, data, dlen);
736040030efSMarcel Holtmann 
737040030efSMarcel Holtmann 	bt_cb(skb)->incoming = 1;
738040030efSMarcel Holtmann 	__net_timestamp(skb);
739040030efSMarcel Holtmann 
740d79f34e3SMarcel Holtmann 	hci_skb_pkt_type(skb) = HCI_EVENT_PKT;
741040030efSMarcel Holtmann 	hci_send_to_sock(hdev, skb);
742040030efSMarcel Holtmann 	kfree_skb(skb);
743040030efSMarcel Holtmann }
744040030efSMarcel Holtmann 
745040030efSMarcel Holtmann void hci_sock_dev_event(struct hci_dev *hdev, int event)
746040030efSMarcel Holtmann {
747040030efSMarcel Holtmann 	BT_DBG("hdev %s event %d", hdev->name, event);
748040030efSMarcel Holtmann 
749cd82e61cSMarcel Holtmann 	if (atomic_read(&monitor_promisc)) {
750cd82e61cSMarcel Holtmann 		struct sk_buff *skb;
751cd82e61cSMarcel Holtmann 
752ed1b28a4SMarcel Holtmann 		/* Send event to monitor */
753cd82e61cSMarcel Holtmann 		skb = create_monitor_event(hdev, event);
754cd82e61cSMarcel Holtmann 		if (skb) {
755c08b1a1dSMarcel Holtmann 			hci_send_to_channel(HCI_CHANNEL_MONITOR, skb,
756c08b1a1dSMarcel Holtmann 					    HCI_SOCK_TRUSTED, NULL);
757cd82e61cSMarcel Holtmann 			kfree_skb(skb);
758cd82e61cSMarcel Holtmann 		}
759cd82e61cSMarcel Holtmann 	}
760cd82e61cSMarcel Holtmann 
761ed1b28a4SMarcel Holtmann 	if (event <= HCI_DEV_DOWN) {
762ed1b28a4SMarcel Holtmann 		struct hci_ev_si_device ev;
763ed1b28a4SMarcel Holtmann 
764040030efSMarcel Holtmann 		/* Send event to sockets */
765040030efSMarcel Holtmann 		ev.event  = event;
766040030efSMarcel Holtmann 		ev.dev_id = hdev->id;
767040030efSMarcel Holtmann 		hci_si_event(NULL, HCI_EV_SI_DEVICE, sizeof(ev), &ev);
768ed1b28a4SMarcel Holtmann 	}
769040030efSMarcel Holtmann 
770040030efSMarcel Holtmann 	if (event == HCI_DEV_UNREG) {
771040030efSMarcel Holtmann 		struct sock *sk;
772040030efSMarcel Holtmann 
773e0448092STetsuo Handa 		/* Wake up sockets using this dead device */
774040030efSMarcel Holtmann 		read_lock(&hci_sk_list.lock);
775b67bfe0dSSasha Levin 		sk_for_each(sk, &hci_sk_list.head) {
776040030efSMarcel Holtmann 			if (hci_pi(sk)->hdev == hdev) {
777040030efSMarcel Holtmann 				sk->sk_err = EPIPE;
778040030efSMarcel Holtmann 				sk->sk_state_change(sk);
779040030efSMarcel Holtmann 			}
780040030efSMarcel Holtmann 		}
781040030efSMarcel Holtmann 		read_unlock(&hci_sk_list.lock);
782040030efSMarcel Holtmann 	}
783040030efSMarcel Holtmann }
784040030efSMarcel Holtmann 
785801c1e8dSJohan Hedberg static struct hci_mgmt_chan *__hci_mgmt_chan_find(unsigned short channel)
786801c1e8dSJohan Hedberg {
787801c1e8dSJohan Hedberg 	struct hci_mgmt_chan *c;
788801c1e8dSJohan Hedberg 
789801c1e8dSJohan Hedberg 	list_for_each_entry(c, &mgmt_chan_list, list) {
790801c1e8dSJohan Hedberg 		if (c->channel == channel)
791801c1e8dSJohan Hedberg 			return c;
792801c1e8dSJohan Hedberg 	}
793801c1e8dSJohan Hedberg 
794801c1e8dSJohan Hedberg 	return NULL;
795801c1e8dSJohan Hedberg }
796801c1e8dSJohan Hedberg 
797801c1e8dSJohan Hedberg static struct hci_mgmt_chan *hci_mgmt_chan_find(unsigned short channel)
798801c1e8dSJohan Hedberg {
799801c1e8dSJohan Hedberg 	struct hci_mgmt_chan *c;
800801c1e8dSJohan Hedberg 
801801c1e8dSJohan Hedberg 	mutex_lock(&mgmt_chan_list_lock);
802801c1e8dSJohan Hedberg 	c = __hci_mgmt_chan_find(channel);
803801c1e8dSJohan Hedberg 	mutex_unlock(&mgmt_chan_list_lock);
804801c1e8dSJohan Hedberg 
805801c1e8dSJohan Hedberg 	return c;
806801c1e8dSJohan Hedberg }
807801c1e8dSJohan Hedberg 
808801c1e8dSJohan Hedberg int hci_mgmt_chan_register(struct hci_mgmt_chan *c)
809801c1e8dSJohan Hedberg {
810801c1e8dSJohan Hedberg 	if (c->channel < HCI_CHANNEL_CONTROL)
811801c1e8dSJohan Hedberg 		return -EINVAL;
812801c1e8dSJohan Hedberg 
813801c1e8dSJohan Hedberg 	mutex_lock(&mgmt_chan_list_lock);
814801c1e8dSJohan Hedberg 	if (__hci_mgmt_chan_find(c->channel)) {
815801c1e8dSJohan Hedberg 		mutex_unlock(&mgmt_chan_list_lock);
816801c1e8dSJohan Hedberg 		return -EALREADY;
817801c1e8dSJohan Hedberg 	}
818801c1e8dSJohan Hedberg 
819801c1e8dSJohan Hedberg 	list_add_tail(&c->list, &mgmt_chan_list);
820801c1e8dSJohan Hedberg 
821801c1e8dSJohan Hedberg 	mutex_unlock(&mgmt_chan_list_lock);
822801c1e8dSJohan Hedberg 
823801c1e8dSJohan Hedberg 	return 0;
824801c1e8dSJohan Hedberg }
825801c1e8dSJohan Hedberg EXPORT_SYMBOL(hci_mgmt_chan_register);
826801c1e8dSJohan Hedberg 
827801c1e8dSJohan Hedberg void hci_mgmt_chan_unregister(struct hci_mgmt_chan *c)
828801c1e8dSJohan Hedberg {
829801c1e8dSJohan Hedberg 	mutex_lock(&mgmt_chan_list_lock);
830801c1e8dSJohan Hedberg 	list_del(&c->list);
831801c1e8dSJohan Hedberg 	mutex_unlock(&mgmt_chan_list_lock);
832801c1e8dSJohan Hedberg }
833801c1e8dSJohan Hedberg EXPORT_SYMBOL(hci_mgmt_chan_unregister);
834801c1e8dSJohan Hedberg 
8351da177e4SLinus Torvalds static int hci_sock_release(struct socket *sock)
8361da177e4SLinus Torvalds {
8371da177e4SLinus Torvalds 	struct sock *sk = sock->sk;
8387b005bd3SMarcel Holtmann 	struct hci_dev *hdev;
839249fa169SMarcel Holtmann 	struct sk_buff *skb;
8401da177e4SLinus Torvalds 
8411da177e4SLinus Torvalds 	BT_DBG("sock %p sk %p", sock, sk);
8421da177e4SLinus Torvalds 
8431da177e4SLinus Torvalds 	if (!sk)
8441da177e4SLinus Torvalds 		return 0;
8451da177e4SLinus Torvalds 
84611eb85ecSDan Carpenter 	lock_sock(sk);
84711eb85ecSDan Carpenter 
84870ecce91SMarcel Holtmann 	switch (hci_pi(sk)->channel) {
84970ecce91SMarcel Holtmann 	case HCI_CHANNEL_MONITOR:
850cd82e61cSMarcel Holtmann 		atomic_dec(&monitor_promisc);
85170ecce91SMarcel Holtmann 		break;
852f81f5b2dSMarcel Holtmann 	case HCI_CHANNEL_RAW:
853aa1638ddSMarcel Holtmann 	case HCI_CHANNEL_USER:
85470ecce91SMarcel Holtmann 	case HCI_CHANNEL_CONTROL:
855249fa169SMarcel Holtmann 		/* Send event to monitor */
856249fa169SMarcel Holtmann 		skb = create_monitor_ctrl_close(sk);
857249fa169SMarcel Holtmann 		if (skb) {
858249fa169SMarcel Holtmann 			hci_send_to_channel(HCI_CHANNEL_MONITOR, skb,
859249fa169SMarcel Holtmann 					    HCI_SOCK_TRUSTED, NULL);
860249fa169SMarcel Holtmann 			kfree_skb(skb);
861249fa169SMarcel Holtmann 		}
862249fa169SMarcel Holtmann 
863df1cb87aSMarcel Holtmann 		hci_sock_free_cookie(sk);
86470ecce91SMarcel Holtmann 		break;
86570ecce91SMarcel Holtmann 	}
866cd82e61cSMarcel Holtmann 
8671da177e4SLinus Torvalds 	bt_sock_unlink(&hci_sk_list, sk);
8681da177e4SLinus Torvalds 
869e20a2e9cSMyungho Jung 	hdev = hci_pi(sk)->hdev;
8701da177e4SLinus Torvalds 	if (hdev) {
87123500189SMarcel Holtmann 		if (hci_pi(sk)->channel == HCI_CHANNEL_USER) {
8729332ef9dSMasahiro Yamada 			/* When releasing a user channel exclusive access,
8736b3cc1dbSSimon Fels 			 * call hci_dev_do_close directly instead of calling
8746b3cc1dbSSimon Fels 			 * hci_dev_close to ensure the exclusive access will
8756b3cc1dbSSimon Fels 			 * be released and the controller brought back down.
8766b3cc1dbSSimon Fels 			 *
8776b3cc1dbSSimon Fels 			 * The checking of HCI_AUTO_OFF is not needed in this
8786b3cc1dbSSimon Fels 			 * case since it will have been cleared already when
8796b3cc1dbSSimon Fels 			 * opening the user channel.
8806b3cc1dbSSimon Fels 			 */
8816b3cc1dbSSimon Fels 			hci_dev_do_close(hdev);
8829380f9eaSLoic Poulain 			hci_dev_clear_flag(hdev, HCI_USER_CHANNEL);
8839380f9eaSLoic Poulain 			mgmt_index_added(hdev);
88423500189SMarcel Holtmann 		}
88523500189SMarcel Holtmann 
8861da177e4SLinus Torvalds 		atomic_dec(&hdev->promisc);
8871da177e4SLinus Torvalds 		hci_dev_put(hdev);
8881da177e4SLinus Torvalds 	}
8891da177e4SLinus Torvalds 
8901da177e4SLinus Torvalds 	sock_orphan(sk);
8911da177e4SLinus Torvalds 
8921da177e4SLinus Torvalds 	skb_queue_purge(&sk->sk_receive_queue);
8931da177e4SLinus Torvalds 	skb_queue_purge(&sk->sk_write_queue);
8941da177e4SLinus Torvalds 
89511eb85ecSDan Carpenter 	release_sock(sk);
8961da177e4SLinus Torvalds 	sock_put(sk);
8971da177e4SLinus Torvalds 	return 0;
8981da177e4SLinus Torvalds }
8991da177e4SLinus Torvalds 
9003d4f9c00SArchie Pusaka static int hci_sock_reject_list_add(struct hci_dev *hdev, void __user *arg)
901f0358568SJohan Hedberg {
902f0358568SJohan Hedberg 	bdaddr_t bdaddr;
9035e762444SAntti Julku 	int err;
904f0358568SJohan Hedberg 
905f0358568SJohan Hedberg 	if (copy_from_user(&bdaddr, arg, sizeof(bdaddr)))
906f0358568SJohan Hedberg 		return -EFAULT;
907f0358568SJohan Hedberg 
90809fd0de5SGustavo F. Padovan 	hci_dev_lock(hdev);
9095e762444SAntti Julku 
9103d4f9c00SArchie Pusaka 	err = hci_bdaddr_list_add(&hdev->reject_list, &bdaddr, BDADDR_BREDR);
9115e762444SAntti Julku 
91209fd0de5SGustavo F. Padovan 	hci_dev_unlock(hdev);
9135e762444SAntti Julku 
9145e762444SAntti Julku 	return err;
915f0358568SJohan Hedberg }
916f0358568SJohan Hedberg 
9173d4f9c00SArchie Pusaka static int hci_sock_reject_list_del(struct hci_dev *hdev, void __user *arg)
918f0358568SJohan Hedberg {
919f0358568SJohan Hedberg 	bdaddr_t bdaddr;
9205e762444SAntti Julku 	int err;
921f0358568SJohan Hedberg 
922f0358568SJohan Hedberg 	if (copy_from_user(&bdaddr, arg, sizeof(bdaddr)))
923f0358568SJohan Hedberg 		return -EFAULT;
924f0358568SJohan Hedberg 
92509fd0de5SGustavo F. Padovan 	hci_dev_lock(hdev);
9265e762444SAntti Julku 
9273d4f9c00SArchie Pusaka 	err = hci_bdaddr_list_del(&hdev->reject_list, &bdaddr, BDADDR_BREDR);
9285e762444SAntti Julku 
92909fd0de5SGustavo F. Padovan 	hci_dev_unlock(hdev);
9305e762444SAntti Julku 
9315e762444SAntti Julku 	return err;
932f0358568SJohan Hedberg }
933f0358568SJohan Hedberg 
9341da177e4SLinus Torvalds /* Ioctls that require bound socket */
9356039aa73SGustavo Padovan static int hci_sock_bound_ioctl(struct sock *sk, unsigned int cmd,
9366039aa73SGustavo Padovan 				unsigned long arg)
9371da177e4SLinus Torvalds {
938e0448092STetsuo Handa 	struct hci_dev *hdev = hci_hdev_from_sock(sk);
9391da177e4SLinus Torvalds 
940e0448092STetsuo Handa 	if (IS_ERR(hdev))
941e0448092STetsuo Handa 		return PTR_ERR(hdev);
9421da177e4SLinus Torvalds 
943d7a5a11dSMarcel Holtmann 	if (hci_dev_test_flag(hdev, HCI_USER_CHANNEL))
9440736cfa8SMarcel Holtmann 		return -EBUSY;
9450736cfa8SMarcel Holtmann 
946d7a5a11dSMarcel Holtmann 	if (hci_dev_test_flag(hdev, HCI_UNCONFIGURED))
947fee746b0SMarcel Holtmann 		return -EOPNOTSUPP;
948fee746b0SMarcel Holtmann 
949ca8bee5dSMarcel Holtmann 	if (hdev->dev_type != HCI_PRIMARY)
9505b69bef5SMarcel Holtmann 		return -EOPNOTSUPP;
9515b69bef5SMarcel Holtmann 
9521da177e4SLinus Torvalds 	switch (cmd) {
9531da177e4SLinus Torvalds 	case HCISETRAW:
9541da177e4SLinus Torvalds 		if (!capable(CAP_NET_ADMIN))
955bf5b30b8SZhao Hongjiang 			return -EPERM;
956db596681SMarcel Holtmann 		return -EOPNOTSUPP;
9571da177e4SLinus Torvalds 
9581da177e4SLinus Torvalds 	case HCIGETCONNINFO:
9591da177e4SLinus Torvalds 		return hci_get_conn_info(hdev, (void __user *)arg);
9601da177e4SLinus Torvalds 
96140be492fSMarcel Holtmann 	case HCIGETAUTHINFO:
96240be492fSMarcel Holtmann 		return hci_get_auth_info(hdev, (void __user *)arg);
96340be492fSMarcel Holtmann 
964f0358568SJohan Hedberg 	case HCIBLOCKADDR:
965f0358568SJohan Hedberg 		if (!capable(CAP_NET_ADMIN))
966bf5b30b8SZhao Hongjiang 			return -EPERM;
9673d4f9c00SArchie Pusaka 		return hci_sock_reject_list_add(hdev, (void __user *)arg);
968f0358568SJohan Hedberg 
969f0358568SJohan Hedberg 	case HCIUNBLOCKADDR:
970f0358568SJohan Hedberg 		if (!capable(CAP_NET_ADMIN))
971bf5b30b8SZhao Hongjiang 			return -EPERM;
9723d4f9c00SArchie Pusaka 		return hci_sock_reject_list_del(hdev, (void __user *)arg);
9730736cfa8SMarcel Holtmann 	}
974f0358568SJohan Hedberg 
975324d36edSMarcel Holtmann 	return -ENOIOCTLCMD;
9761da177e4SLinus Torvalds }
9771da177e4SLinus Torvalds 
9788fc9ced3SGustavo Padovan static int hci_sock_ioctl(struct socket *sock, unsigned int cmd,
9798fc9ced3SGustavo Padovan 			  unsigned long arg)
9801da177e4SLinus Torvalds {
9811da177e4SLinus Torvalds 	void __user *argp = (void __user *)arg;
9820736cfa8SMarcel Holtmann 	struct sock *sk = sock->sk;
9831da177e4SLinus Torvalds 	int err;
9841da177e4SLinus Torvalds 
9851da177e4SLinus Torvalds 	BT_DBG("cmd %x arg %lx", cmd, arg);
9861da177e4SLinus Torvalds 
987c1c4f956SMarcel Holtmann 	lock_sock(sk);
988c1c4f956SMarcel Holtmann 
989c1c4f956SMarcel Holtmann 	if (hci_pi(sk)->channel != HCI_CHANNEL_RAW) {
990c1c4f956SMarcel Holtmann 		err = -EBADFD;
991c1c4f956SMarcel Holtmann 		goto done;
992c1c4f956SMarcel Holtmann 	}
993c1c4f956SMarcel Holtmann 
994f81f5b2dSMarcel Holtmann 	/* When calling an ioctl on an unbound raw socket, then ensure
995f81f5b2dSMarcel Holtmann 	 * that the monitor gets informed. Ensure that the resulting event
996f81f5b2dSMarcel Holtmann 	 * is only send once by checking if the cookie exists or not. The
997f81f5b2dSMarcel Holtmann 	 * socket cookie will be only ever generated once for the lifetime
998f81f5b2dSMarcel Holtmann 	 * of a given socket.
999f81f5b2dSMarcel Holtmann 	 */
1000f81f5b2dSMarcel Holtmann 	if (hci_sock_gen_cookie(sk)) {
1001f81f5b2dSMarcel Holtmann 		struct sk_buff *skb;
1002f81f5b2dSMarcel Holtmann 
1003f81f5b2dSMarcel Holtmann 		if (capable(CAP_NET_ADMIN))
1004f81f5b2dSMarcel Holtmann 			hci_sock_set_flag(sk, HCI_SOCK_TRUSTED);
1005f81f5b2dSMarcel Holtmann 
1006f81f5b2dSMarcel Holtmann 		/* Send event to monitor */
1007f81f5b2dSMarcel Holtmann 		skb = create_monitor_ctrl_open(sk);
1008f81f5b2dSMarcel Holtmann 		if (skb) {
1009f81f5b2dSMarcel Holtmann 			hci_send_to_channel(HCI_CHANNEL_MONITOR, skb,
1010f81f5b2dSMarcel Holtmann 					    HCI_SOCK_TRUSTED, NULL);
1011f81f5b2dSMarcel Holtmann 			kfree_skb(skb);
1012f81f5b2dSMarcel Holtmann 		}
1013f81f5b2dSMarcel Holtmann 	}
1014f81f5b2dSMarcel Holtmann 
1015c1c4f956SMarcel Holtmann 	release_sock(sk);
1016c1c4f956SMarcel Holtmann 
10171da177e4SLinus Torvalds 	switch (cmd) {
10181da177e4SLinus Torvalds 	case HCIGETDEVLIST:
10191da177e4SLinus Torvalds 		return hci_get_dev_list(argp);
10201da177e4SLinus Torvalds 
10211da177e4SLinus Torvalds 	case HCIGETDEVINFO:
10221da177e4SLinus Torvalds 		return hci_get_dev_info(argp);
10231da177e4SLinus Torvalds 
10241da177e4SLinus Torvalds 	case HCIGETCONNLIST:
10251da177e4SLinus Torvalds 		return hci_get_conn_list(argp);
10261da177e4SLinus Torvalds 
10271da177e4SLinus Torvalds 	case HCIDEVUP:
10281da177e4SLinus Torvalds 		if (!capable(CAP_NET_ADMIN))
1029bf5b30b8SZhao Hongjiang 			return -EPERM;
10301da177e4SLinus Torvalds 		return hci_dev_open(arg);
10311da177e4SLinus Torvalds 
10321da177e4SLinus Torvalds 	case HCIDEVDOWN:
10331da177e4SLinus Torvalds 		if (!capable(CAP_NET_ADMIN))
1034bf5b30b8SZhao Hongjiang 			return -EPERM;
10351da177e4SLinus Torvalds 		return hci_dev_close(arg);
10361da177e4SLinus Torvalds 
10371da177e4SLinus Torvalds 	case HCIDEVRESET:
10381da177e4SLinus Torvalds 		if (!capable(CAP_NET_ADMIN))
1039bf5b30b8SZhao Hongjiang 			return -EPERM;
10401da177e4SLinus Torvalds 		return hci_dev_reset(arg);
10411da177e4SLinus Torvalds 
10421da177e4SLinus Torvalds 	case HCIDEVRESTAT:
10431da177e4SLinus Torvalds 		if (!capable(CAP_NET_ADMIN))
1044bf5b30b8SZhao Hongjiang 			return -EPERM;
10451da177e4SLinus Torvalds 		return hci_dev_reset_stat(arg);
10461da177e4SLinus Torvalds 
10471da177e4SLinus Torvalds 	case HCISETSCAN:
10481da177e4SLinus Torvalds 	case HCISETAUTH:
10491da177e4SLinus Torvalds 	case HCISETENCRYPT:
10501da177e4SLinus Torvalds 	case HCISETPTYPE:
10511da177e4SLinus Torvalds 	case HCISETLINKPOL:
10521da177e4SLinus Torvalds 	case HCISETLINKMODE:
10531da177e4SLinus Torvalds 	case HCISETACLMTU:
10541da177e4SLinus Torvalds 	case HCISETSCOMTU:
10551da177e4SLinus Torvalds 		if (!capable(CAP_NET_ADMIN))
1056bf5b30b8SZhao Hongjiang 			return -EPERM;
10571da177e4SLinus Torvalds 		return hci_dev_cmd(cmd, argp);
10581da177e4SLinus Torvalds 
10591da177e4SLinus Torvalds 	case HCIINQUIRY:
10601da177e4SLinus Torvalds 		return hci_inquiry(argp);
1061c1c4f956SMarcel Holtmann 	}
10621da177e4SLinus Torvalds 
10631da177e4SLinus Torvalds 	lock_sock(sk);
1064c1c4f956SMarcel Holtmann 
10651da177e4SLinus Torvalds 	err = hci_sock_bound_ioctl(sk, cmd, arg);
1066c1c4f956SMarcel Holtmann 
1067c1c4f956SMarcel Holtmann done:
10681da177e4SLinus Torvalds 	release_sock(sk);
10691da177e4SLinus Torvalds 	return err;
10701da177e4SLinus Torvalds }
10711da177e4SLinus Torvalds 
10727a6038b3SArnd Bergmann #ifdef CONFIG_COMPAT
10737a6038b3SArnd Bergmann static int hci_sock_compat_ioctl(struct socket *sock, unsigned int cmd,
10747a6038b3SArnd Bergmann 				 unsigned long arg)
10757a6038b3SArnd Bergmann {
10767a6038b3SArnd Bergmann 	switch (cmd) {
10777a6038b3SArnd Bergmann 	case HCIDEVUP:
10787a6038b3SArnd Bergmann 	case HCIDEVDOWN:
10797a6038b3SArnd Bergmann 	case HCIDEVRESET:
10807a6038b3SArnd Bergmann 	case HCIDEVRESTAT:
10817a6038b3SArnd Bergmann 		return hci_sock_ioctl(sock, cmd, arg);
10827a6038b3SArnd Bergmann 	}
10837a6038b3SArnd Bergmann 
10847a6038b3SArnd Bergmann 	return hci_sock_ioctl(sock, cmd, (unsigned long)compat_ptr(arg));
10857a6038b3SArnd Bergmann }
10867a6038b3SArnd Bergmann #endif
10877a6038b3SArnd Bergmann 
10888fc9ced3SGustavo Padovan static int hci_sock_bind(struct socket *sock, struct sockaddr *addr,
10898fc9ced3SGustavo Padovan 			 int addr_len)
10901da177e4SLinus Torvalds {
10910381101fSJohan Hedberg 	struct sockaddr_hci haddr;
10921da177e4SLinus Torvalds 	struct sock *sk = sock->sk;
10931da177e4SLinus Torvalds 	struct hci_dev *hdev = NULL;
1094f4cdbb3fSMarcel Holtmann 	struct sk_buff *skb;
10950381101fSJohan Hedberg 	int len, err = 0;
10961da177e4SLinus Torvalds 
10971da177e4SLinus Torvalds 	BT_DBG("sock %p sk %p", sock, sk);
10981da177e4SLinus Torvalds 
10990381101fSJohan Hedberg 	if (!addr)
11000381101fSJohan Hedberg 		return -EINVAL;
11010381101fSJohan Hedberg 
11020381101fSJohan Hedberg 	memset(&haddr, 0, sizeof(haddr));
11030381101fSJohan Hedberg 	len = min_t(unsigned int, sizeof(haddr), addr_len);
11040381101fSJohan Hedberg 	memcpy(&haddr, addr, len);
11050381101fSJohan Hedberg 
11060381101fSJohan Hedberg 	if (haddr.hci_family != AF_BLUETOOTH)
11070381101fSJohan Hedberg 		return -EINVAL;
11080381101fSJohan Hedberg 
11091da177e4SLinus Torvalds 	lock_sock(sk);
11101da177e4SLinus Torvalds 
1111e0448092STetsuo Handa 	/* Allow detaching from dead device and attaching to alive device, if
1112e0448092STetsuo Handa 	 * the caller wants to re-bind (instead of close) this socket in
1113e0448092STetsuo Handa 	 * response to hci_sock_dev_event(HCI_DEV_UNREG) notification.
1114e0448092STetsuo Handa 	 */
1115e0448092STetsuo Handa 	hdev = hci_pi(sk)->hdev;
1116e0448092STetsuo Handa 	if (hdev && hci_dev_test_flag(hdev, HCI_UNREGISTER)) {
1117e0448092STetsuo Handa 		hci_pi(sk)->hdev = NULL;
1118e0448092STetsuo Handa 		sk->sk_state = BT_OPEN;
1119e0448092STetsuo Handa 		hci_dev_put(hdev);
1120e0448092STetsuo Handa 	}
1121e0448092STetsuo Handa 	hdev = NULL;
1122e0448092STetsuo Handa 
11237cc2ade2SMarcel Holtmann 	if (sk->sk_state == BT_BOUND) {
11247cc2ade2SMarcel Holtmann 		err = -EALREADY;
11257cc2ade2SMarcel Holtmann 		goto done;
11267cc2ade2SMarcel Holtmann 	}
11277cc2ade2SMarcel Holtmann 
11287cc2ade2SMarcel Holtmann 	switch (haddr.hci_channel) {
11297cc2ade2SMarcel Holtmann 	case HCI_CHANNEL_RAW:
11307cc2ade2SMarcel Holtmann 		if (hci_pi(sk)->hdev) {
11311da177e4SLinus Torvalds 			err = -EALREADY;
11321da177e4SLinus Torvalds 			goto done;
11331da177e4SLinus Torvalds 		}
11341da177e4SLinus Torvalds 
11350381101fSJohan Hedberg 		if (haddr.hci_dev != HCI_DEV_NONE) {
11360381101fSJohan Hedberg 			hdev = hci_dev_get(haddr.hci_dev);
113770f23020SAndrei Emeltchenko 			if (!hdev) {
11381da177e4SLinus Torvalds 				err = -ENODEV;
11391da177e4SLinus Torvalds 				goto done;
11401da177e4SLinus Torvalds 			}
11411da177e4SLinus Torvalds 
11421da177e4SLinus Torvalds 			atomic_inc(&hdev->promisc);
11431da177e4SLinus Torvalds 		}
11441da177e4SLinus Torvalds 
11455a6d2cf5SMarcel Holtmann 		hci_pi(sk)->channel = haddr.hci_channel;
1146f81f5b2dSMarcel Holtmann 
1147f4cdbb3fSMarcel Holtmann 		if (!hci_sock_gen_cookie(sk)) {
1148f4cdbb3fSMarcel Holtmann 			/* In the case when a cookie has already been assigned,
1149f4cdbb3fSMarcel Holtmann 			 * then there has been already an ioctl issued against
115091641b79SZheng Yongjun 			 * an unbound socket and with that triggered an open
1151f4cdbb3fSMarcel Holtmann 			 * notification. Send a close notification first to
1152f4cdbb3fSMarcel Holtmann 			 * allow the state transition to bounded.
1153f81f5b2dSMarcel Holtmann 			 */
1154f4cdbb3fSMarcel Holtmann 			skb = create_monitor_ctrl_close(sk);
1155f4cdbb3fSMarcel Holtmann 			if (skb) {
1156f4cdbb3fSMarcel Holtmann 				hci_send_to_channel(HCI_CHANNEL_MONITOR, skb,
1157f4cdbb3fSMarcel Holtmann 						    HCI_SOCK_TRUSTED, NULL);
1158f4cdbb3fSMarcel Holtmann 				kfree_skb(skb);
1159f4cdbb3fSMarcel Holtmann 			}
1160f4cdbb3fSMarcel Holtmann 		}
1161f81f5b2dSMarcel Holtmann 
1162f81f5b2dSMarcel Holtmann 		if (capable(CAP_NET_ADMIN))
1163f81f5b2dSMarcel Holtmann 			hci_sock_set_flag(sk, HCI_SOCK_TRUSTED);
1164f81f5b2dSMarcel Holtmann 
1165f4cdbb3fSMarcel Holtmann 		hci_pi(sk)->hdev = hdev;
1166f4cdbb3fSMarcel Holtmann 
1167f81f5b2dSMarcel Holtmann 		/* Send event to monitor */
1168f81f5b2dSMarcel Holtmann 		skb = create_monitor_ctrl_open(sk);
1169f81f5b2dSMarcel Holtmann 		if (skb) {
1170f81f5b2dSMarcel Holtmann 			hci_send_to_channel(HCI_CHANNEL_MONITOR, skb,
1171f81f5b2dSMarcel Holtmann 					    HCI_SOCK_TRUSTED, NULL);
1172f81f5b2dSMarcel Holtmann 			kfree_skb(skb);
1173f81f5b2dSMarcel Holtmann 		}
11747cc2ade2SMarcel Holtmann 		break;
11757cc2ade2SMarcel Holtmann 
117623500189SMarcel Holtmann 	case HCI_CHANNEL_USER:
117723500189SMarcel Holtmann 		if (hci_pi(sk)->hdev) {
117823500189SMarcel Holtmann 			err = -EALREADY;
117923500189SMarcel Holtmann 			goto done;
118023500189SMarcel Holtmann 		}
118123500189SMarcel Holtmann 
118223500189SMarcel Holtmann 		if (haddr.hci_dev == HCI_DEV_NONE) {
118323500189SMarcel Holtmann 			err = -EINVAL;
118423500189SMarcel Holtmann 			goto done;
118523500189SMarcel Holtmann 		}
118623500189SMarcel Holtmann 
118710a8b86fSMarcel Holtmann 		if (!capable(CAP_NET_ADMIN)) {
118823500189SMarcel Holtmann 			err = -EPERM;
118923500189SMarcel Holtmann 			goto done;
119023500189SMarcel Holtmann 		}
119123500189SMarcel Holtmann 
119223500189SMarcel Holtmann 		hdev = hci_dev_get(haddr.hci_dev);
119323500189SMarcel Holtmann 		if (!hdev) {
119423500189SMarcel Holtmann 			err = -ENODEV;
119523500189SMarcel Holtmann 			goto done;
119623500189SMarcel Holtmann 		}
119723500189SMarcel Holtmann 
1198781f899fSMarcel Holtmann 		if (test_bit(HCI_INIT, &hdev->flags) ||
1199d7a5a11dSMarcel Holtmann 		    hci_dev_test_flag(hdev, HCI_SETUP) ||
1200781f899fSMarcel Holtmann 		    hci_dev_test_flag(hdev, HCI_CONFIG) ||
1201781f899fSMarcel Holtmann 		    (!hci_dev_test_flag(hdev, HCI_AUTO_OFF) &&
1202781f899fSMarcel Holtmann 		     test_bit(HCI_UP, &hdev->flags))) {
120323500189SMarcel Holtmann 			err = -EBUSY;
120423500189SMarcel Holtmann 			hci_dev_put(hdev);
120523500189SMarcel Holtmann 			goto done;
120623500189SMarcel Holtmann 		}
120723500189SMarcel Holtmann 
1208238be788SMarcel Holtmann 		if (hci_dev_test_and_set_flag(hdev, HCI_USER_CHANNEL)) {
120923500189SMarcel Holtmann 			err = -EUSERS;
121023500189SMarcel Holtmann 			hci_dev_put(hdev);
121123500189SMarcel Holtmann 			goto done;
121223500189SMarcel Holtmann 		}
121323500189SMarcel Holtmann 
121423500189SMarcel Holtmann 		mgmt_index_removed(hdev);
121523500189SMarcel Holtmann 
121623500189SMarcel Holtmann 		err = hci_dev_open(hdev->id);
121723500189SMarcel Holtmann 		if (err) {
1218781f899fSMarcel Holtmann 			if (err == -EALREADY) {
1219781f899fSMarcel Holtmann 				/* In case the transport is already up and
1220781f899fSMarcel Holtmann 				 * running, clear the error here.
1221781f899fSMarcel Holtmann 				 *
12229332ef9dSMasahiro Yamada 				 * This can happen when opening a user
1223781f899fSMarcel Holtmann 				 * channel and HCI_AUTO_OFF grace period
1224781f899fSMarcel Holtmann 				 * is still active.
1225781f899fSMarcel Holtmann 				 */
1226781f899fSMarcel Holtmann 				err = 0;
1227781f899fSMarcel Holtmann 			} else {
1228a358dc11SMarcel Holtmann 				hci_dev_clear_flag(hdev, HCI_USER_CHANNEL);
1229c6521401SMarcel Holtmann 				mgmt_index_added(hdev);
123023500189SMarcel Holtmann 				hci_dev_put(hdev);
123123500189SMarcel Holtmann 				goto done;
123223500189SMarcel Holtmann 			}
1233781f899fSMarcel Holtmann 		}
123423500189SMarcel Holtmann 
12355a6d2cf5SMarcel Holtmann 		hci_pi(sk)->channel = haddr.hci_channel;
1236aa1638ddSMarcel Holtmann 
1237aa1638ddSMarcel Holtmann 		if (!hci_sock_gen_cookie(sk)) {
1238aa1638ddSMarcel Holtmann 			/* In the case when a cookie has already been assigned,
1239aa1638ddSMarcel Holtmann 			 * this socket will transition from a raw socket into
12409332ef9dSMasahiro Yamada 			 * a user channel socket. For a clean transition, send
1241aa1638ddSMarcel Holtmann 			 * the close notification first.
1242aa1638ddSMarcel Holtmann 			 */
1243aa1638ddSMarcel Holtmann 			skb = create_monitor_ctrl_close(sk);
1244aa1638ddSMarcel Holtmann 			if (skb) {
1245aa1638ddSMarcel Holtmann 				hci_send_to_channel(HCI_CHANNEL_MONITOR, skb,
1246aa1638ddSMarcel Holtmann 						    HCI_SOCK_TRUSTED, NULL);
1247aa1638ddSMarcel Holtmann 				kfree_skb(skb);
1248aa1638ddSMarcel Holtmann 			}
1249aa1638ddSMarcel Holtmann 		}
1250aa1638ddSMarcel Holtmann 
1251aa1638ddSMarcel Holtmann 		/* The user channel is restricted to CAP_NET_ADMIN
1252aa1638ddSMarcel Holtmann 		 * capabilities and with that implicitly trusted.
1253aa1638ddSMarcel Holtmann 		 */
1254aa1638ddSMarcel Holtmann 		hci_sock_set_flag(sk, HCI_SOCK_TRUSTED);
1255aa1638ddSMarcel Holtmann 
125623500189SMarcel Holtmann 		hci_pi(sk)->hdev = hdev;
12575a6d2cf5SMarcel Holtmann 
1258aa1638ddSMarcel Holtmann 		/* Send event to monitor */
1259aa1638ddSMarcel Holtmann 		skb = create_monitor_ctrl_open(sk);
1260aa1638ddSMarcel Holtmann 		if (skb) {
1261aa1638ddSMarcel Holtmann 			hci_send_to_channel(HCI_CHANNEL_MONITOR, skb,
1262aa1638ddSMarcel Holtmann 					    HCI_SOCK_TRUSTED, NULL);
1263aa1638ddSMarcel Holtmann 			kfree_skb(skb);
1264aa1638ddSMarcel Holtmann 		}
1265aa1638ddSMarcel Holtmann 
12665a6d2cf5SMarcel Holtmann 		atomic_inc(&hdev->promisc);
126723500189SMarcel Holtmann 		break;
126823500189SMarcel Holtmann 
1269cd82e61cSMarcel Holtmann 	case HCI_CHANNEL_MONITOR:
1270cd82e61cSMarcel Holtmann 		if (haddr.hci_dev != HCI_DEV_NONE) {
1271cd82e61cSMarcel Holtmann 			err = -EINVAL;
1272cd82e61cSMarcel Holtmann 			goto done;
1273cd82e61cSMarcel Holtmann 		}
1274cd82e61cSMarcel Holtmann 
1275cd82e61cSMarcel Holtmann 		if (!capable(CAP_NET_RAW)) {
1276cd82e61cSMarcel Holtmann 			err = -EPERM;
1277cd82e61cSMarcel Holtmann 			goto done;
1278cd82e61cSMarcel Holtmann 		}
1279cd82e61cSMarcel Holtmann 
12805a6d2cf5SMarcel Holtmann 		hci_pi(sk)->channel = haddr.hci_channel;
12815a6d2cf5SMarcel Holtmann 
128250ebc055SMarcel Holtmann 		/* The monitor interface is restricted to CAP_NET_RAW
128350ebc055SMarcel Holtmann 		 * capabilities and with that implicitly trusted.
128450ebc055SMarcel Holtmann 		 */
128550ebc055SMarcel Holtmann 		hci_sock_set_flag(sk, HCI_SOCK_TRUSTED);
128650ebc055SMarcel Holtmann 
1287787b306cSJohannes Berg 		send_monitor_note(sk, "Linux version %s (%s)",
1288787b306cSJohannes Berg 				  init_utsname()->release,
1289787b306cSJohannes Berg 				  init_utsname()->machine);
12909e8305b3SMarcel Holtmann 		send_monitor_note(sk, "Bluetooth subsystem version %u.%u",
12919e8305b3SMarcel Holtmann 				  BT_SUBSYS_VERSION, BT_SUBSYS_REVISION);
1292cd82e61cSMarcel Holtmann 		send_monitor_replay(sk);
1293249fa169SMarcel Holtmann 		send_monitor_control_replay(sk);
1294cd82e61cSMarcel Holtmann 
1295cd82e61cSMarcel Holtmann 		atomic_inc(&monitor_promisc);
1296cd82e61cSMarcel Holtmann 		break;
1297cd82e61cSMarcel Holtmann 
1298ac714949SMarcel Holtmann 	case HCI_CHANNEL_LOGGING:
1299ac714949SMarcel Holtmann 		if (haddr.hci_dev != HCI_DEV_NONE) {
1300ac714949SMarcel Holtmann 			err = -EINVAL;
1301ac714949SMarcel Holtmann 			goto done;
1302ac714949SMarcel Holtmann 		}
1303ac714949SMarcel Holtmann 
1304ac714949SMarcel Holtmann 		if (!capable(CAP_NET_ADMIN)) {
1305ac714949SMarcel Holtmann 			err = -EPERM;
1306ac714949SMarcel Holtmann 			goto done;
1307ac714949SMarcel Holtmann 		}
13085a6d2cf5SMarcel Holtmann 
13095a6d2cf5SMarcel Holtmann 		hci_pi(sk)->channel = haddr.hci_channel;
1310ac714949SMarcel Holtmann 		break;
1311ac714949SMarcel Holtmann 
13127cc2ade2SMarcel Holtmann 	default:
1313801c1e8dSJohan Hedberg 		if (!hci_mgmt_chan_find(haddr.hci_channel)) {
13147cc2ade2SMarcel Holtmann 			err = -EINVAL;
13157cc2ade2SMarcel Holtmann 			goto done;
13167cc2ade2SMarcel Holtmann 		}
13177cc2ade2SMarcel Holtmann 
1318801c1e8dSJohan Hedberg 		if (haddr.hci_dev != HCI_DEV_NONE) {
1319801c1e8dSJohan Hedberg 			err = -EINVAL;
1320801c1e8dSJohan Hedberg 			goto done;
1321801c1e8dSJohan Hedberg 		}
1322801c1e8dSJohan Hedberg 
13231195fbb8SMarcel Holtmann 		/* Users with CAP_NET_ADMIN capabilities are allowed
13241195fbb8SMarcel Holtmann 		 * access to all management commands and events. For
13251195fbb8SMarcel Holtmann 		 * untrusted users the interface is restricted and
13261195fbb8SMarcel Holtmann 		 * also only untrusted events are sent.
132750ebc055SMarcel Holtmann 		 */
13281195fbb8SMarcel Holtmann 		if (capable(CAP_NET_ADMIN))
132950ebc055SMarcel Holtmann 			hci_sock_set_flag(sk, HCI_SOCK_TRUSTED);
133050ebc055SMarcel Holtmann 
13315a6d2cf5SMarcel Holtmann 		hci_pi(sk)->channel = haddr.hci_channel;
13325a6d2cf5SMarcel Holtmann 
1333f9207338SMarcel Holtmann 		/* At the moment the index and unconfigured index events
1334f9207338SMarcel Holtmann 		 * are enabled unconditionally. Setting them on each
1335f9207338SMarcel Holtmann 		 * socket when binding keeps this functionality. They
1336f9207338SMarcel Holtmann 		 * however might be cleared later and then sending of these
1337f9207338SMarcel Holtmann 		 * events will be disabled, but that is then intentional.
1338f6b7712eSMarcel Holtmann 		 *
1339f6b7712eSMarcel Holtmann 		 * This also enables generic events that are safe to be
1340f6b7712eSMarcel Holtmann 		 * received by untrusted users. Example for such events
1341f6b7712eSMarcel Holtmann 		 * are changes to settings, class of device, name etc.
1342f9207338SMarcel Holtmann 		 */
13435a6d2cf5SMarcel Holtmann 		if (hci_pi(sk)->channel == HCI_CHANNEL_CONTROL) {
1344f4cdbb3fSMarcel Holtmann 			if (!hci_sock_gen_cookie(sk)) {
1345f4cdbb3fSMarcel Holtmann 				/* In the case when a cookie has already been
134691641b79SZheng Yongjun 				 * assigned, this socket will transition from
1347f4cdbb3fSMarcel Holtmann 				 * a raw socket into a control socket. To
134891641b79SZheng Yongjun 				 * allow for a clean transition, send the
1349f4cdbb3fSMarcel Holtmann 				 * close notification first.
1350f4cdbb3fSMarcel Holtmann 				 */
1351f4cdbb3fSMarcel Holtmann 				skb = create_monitor_ctrl_close(sk);
1352f4cdbb3fSMarcel Holtmann 				if (skb) {
1353f4cdbb3fSMarcel Holtmann 					hci_send_to_channel(HCI_CHANNEL_MONITOR, skb,
1354f4cdbb3fSMarcel Holtmann 							    HCI_SOCK_TRUSTED, NULL);
1355f4cdbb3fSMarcel Holtmann 					kfree_skb(skb);
1356f4cdbb3fSMarcel Holtmann 				}
1357f4cdbb3fSMarcel Holtmann 			}
135870ecce91SMarcel Holtmann 
1359249fa169SMarcel Holtmann 			/* Send event to monitor */
1360249fa169SMarcel Holtmann 			skb = create_monitor_ctrl_open(sk);
1361249fa169SMarcel Holtmann 			if (skb) {
1362249fa169SMarcel Holtmann 				hci_send_to_channel(HCI_CHANNEL_MONITOR, skb,
1363249fa169SMarcel Holtmann 						    HCI_SOCK_TRUSTED, NULL);
1364249fa169SMarcel Holtmann 				kfree_skb(skb);
1365249fa169SMarcel Holtmann 			}
1366249fa169SMarcel Holtmann 
1367f9207338SMarcel Holtmann 			hci_sock_set_flag(sk, HCI_MGMT_INDEX_EVENTS);
1368f9207338SMarcel Holtmann 			hci_sock_set_flag(sk, HCI_MGMT_UNCONF_INDEX_EVENTS);
13695504c3a3SMarcel Holtmann 			hci_sock_set_flag(sk, HCI_MGMT_OPTION_EVENTS);
13705504c3a3SMarcel Holtmann 			hci_sock_set_flag(sk, HCI_MGMT_SETTING_EVENTS);
13715504c3a3SMarcel Holtmann 			hci_sock_set_flag(sk, HCI_MGMT_DEV_CLASS_EVENTS);
13725504c3a3SMarcel Holtmann 			hci_sock_set_flag(sk, HCI_MGMT_LOCAL_NAME_EVENTS);
1373f9207338SMarcel Holtmann 		}
1374801c1e8dSJohan Hedberg 		break;
1375801c1e8dSJohan Hedberg 	}
1376801c1e8dSJohan Hedberg 
13771da177e4SLinus Torvalds 	sk->sk_state = BT_BOUND;
13781da177e4SLinus Torvalds 
13791da177e4SLinus Torvalds done:
13801da177e4SLinus Torvalds 	release_sock(sk);
13811da177e4SLinus Torvalds 	return err;
13821da177e4SLinus Torvalds }
13831da177e4SLinus Torvalds 
13848fc9ced3SGustavo Padovan static int hci_sock_getname(struct socket *sock, struct sockaddr *addr,
13859b2c45d4SDenys Vlasenko 			    int peer)
13861da177e4SLinus Torvalds {
13871da177e4SLinus Torvalds 	struct sockaddr_hci *haddr = (struct sockaddr_hci *)addr;
13881da177e4SLinus Torvalds 	struct sock *sk = sock->sk;
13899d4b68b2SMarcel Holtmann 	struct hci_dev *hdev;
13909d4b68b2SMarcel Holtmann 	int err = 0;
13911da177e4SLinus Torvalds 
13921da177e4SLinus Torvalds 	BT_DBG("sock %p sk %p", sock, sk);
13931da177e4SLinus Torvalds 
139406f43cbcSMarcel Holtmann 	if (peer)
139506f43cbcSMarcel Holtmann 		return -EOPNOTSUPP;
139606f43cbcSMarcel Holtmann 
13971da177e4SLinus Torvalds 	lock_sock(sk);
13981da177e4SLinus Torvalds 
1399e0448092STetsuo Handa 	hdev = hci_hdev_from_sock(sk);
1400e0448092STetsuo Handa 	if (IS_ERR(hdev)) {
1401e0448092STetsuo Handa 		err = PTR_ERR(hdev);
14029d4b68b2SMarcel Holtmann 		goto done;
14039d4b68b2SMarcel Holtmann 	}
14049d4b68b2SMarcel Holtmann 
14051da177e4SLinus Torvalds 	haddr->hci_family = AF_BLUETOOTH;
14067b005bd3SMarcel Holtmann 	haddr->hci_dev    = hdev->id;
14079d4b68b2SMarcel Holtmann 	haddr->hci_channel= hci_pi(sk)->channel;
14089b2c45d4SDenys Vlasenko 	err = sizeof(*haddr);
14091da177e4SLinus Torvalds 
14109d4b68b2SMarcel Holtmann done:
14111da177e4SLinus Torvalds 	release_sock(sk);
14129d4b68b2SMarcel Holtmann 	return err;
14131da177e4SLinus Torvalds }
14141da177e4SLinus Torvalds 
14156039aa73SGustavo Padovan static void hci_sock_cmsg(struct sock *sk, struct msghdr *msg,
14166039aa73SGustavo Padovan 			  struct sk_buff *skb)
14171da177e4SLinus Torvalds {
141832929e1fSAlain Michaud 	__u8 mask = hci_pi(sk)->cmsg_mask;
14191da177e4SLinus Torvalds 
14200d48d939SMarcel Holtmann 	if (mask & HCI_CMSG_DIR) {
14210d48d939SMarcel Holtmann 		int incoming = bt_cb(skb)->incoming;
14228fc9ced3SGustavo Padovan 		put_cmsg(msg, SOL_HCI, HCI_CMSG_DIR, sizeof(incoming),
14238fc9ced3SGustavo Padovan 			 &incoming);
14240d48d939SMarcel Holtmann 	}
14251da177e4SLinus Torvalds 
1426a61bbcf2SPatrick McHardy 	if (mask & HCI_CMSG_TSTAMP) {
1427f6e623a6SJohann Felix Soden #ifdef CONFIG_COMPAT
142813c6ee2aSDeepa Dinamani 		struct old_timeval32 ctv;
1429f6e623a6SJohann Felix Soden #endif
143013c6ee2aSDeepa Dinamani 		struct __kernel_old_timeval tv;
1431767c5eb5SMarcel Holtmann 		void *data;
1432767c5eb5SMarcel Holtmann 		int len;
1433a61bbcf2SPatrick McHardy 
1434a61bbcf2SPatrick McHardy 		skb_get_timestamp(skb, &tv);
1435767c5eb5SMarcel Holtmann 
14361da97f83SDavid S. Miller 		data = &tv;
14371da97f83SDavid S. Miller 		len = sizeof(tv);
14381da97f83SDavid S. Miller #ifdef CONFIG_COMPAT
1439da88cea1SH. J. Lu 		if (!COMPAT_USE_64BIT_TIME &&
1440da88cea1SH. J. Lu 		    (msg->msg_flags & MSG_CMSG_COMPAT)) {
1441767c5eb5SMarcel Holtmann 			ctv.tv_sec = tv.tv_sec;
1442767c5eb5SMarcel Holtmann 			ctv.tv_usec = tv.tv_usec;
1443767c5eb5SMarcel Holtmann 			data = &ctv;
1444767c5eb5SMarcel Holtmann 			len = sizeof(ctv);
1445767c5eb5SMarcel Holtmann 		}
14461da97f83SDavid S. Miller #endif
1447767c5eb5SMarcel Holtmann 
1448767c5eb5SMarcel Holtmann 		put_cmsg(msg, SOL_HCI, HCI_CMSG_TSTAMP, len, data);
1449a61bbcf2SPatrick McHardy 	}
14501da177e4SLinus Torvalds }
14511da177e4SLinus Torvalds 
14528528d3f7SMarcel Holtmann static int hci_sock_recvmsg(struct socket *sock, struct msghdr *msg,
14538528d3f7SMarcel Holtmann 			    size_t len, int flags)
14541da177e4SLinus Torvalds {
14551da177e4SLinus Torvalds 	int noblock = flags & MSG_DONTWAIT;
14561da177e4SLinus Torvalds 	struct sock *sk = sock->sk;
14571da177e4SLinus Torvalds 	struct sk_buff *skb;
14581da177e4SLinus Torvalds 	int copied, err;
145983871f8cSDenis Kenzior 	unsigned int skblen;
14601da177e4SLinus Torvalds 
14611da177e4SLinus Torvalds 	BT_DBG("sock %p, sk %p", sock, sk);
14621da177e4SLinus Torvalds 
1463d94a6104SMarcel Holtmann 	if (flags & MSG_OOB)
14641da177e4SLinus Torvalds 		return -EOPNOTSUPP;
14651da177e4SLinus Torvalds 
1466ac714949SMarcel Holtmann 	if (hci_pi(sk)->channel == HCI_CHANNEL_LOGGING)
1467ac714949SMarcel Holtmann 		return -EOPNOTSUPP;
1468ac714949SMarcel Holtmann 
14691da177e4SLinus Torvalds 	if (sk->sk_state == BT_CLOSED)
14701da177e4SLinus Torvalds 		return 0;
14711da177e4SLinus Torvalds 
147270f23020SAndrei Emeltchenko 	skb = skb_recv_datagram(sk, flags, noblock, &err);
147370f23020SAndrei Emeltchenko 	if (!skb)
14741da177e4SLinus Torvalds 		return err;
14751da177e4SLinus Torvalds 
147683871f8cSDenis Kenzior 	skblen = skb->len;
14771da177e4SLinus Torvalds 	copied = skb->len;
14781da177e4SLinus Torvalds 	if (len < copied) {
14791da177e4SLinus Torvalds 		msg->msg_flags |= MSG_TRUNC;
14801da177e4SLinus Torvalds 		copied = len;
14811da177e4SLinus Torvalds 	}
14821da177e4SLinus Torvalds 
1483badff6d0SArnaldo Carvalho de Melo 	skb_reset_transport_header(skb);
148451f3d02bSDavid S. Miller 	err = skb_copy_datagram_msg(skb, 0, msg, copied);
14851da177e4SLinus Torvalds 
14863a208627SMarcel Holtmann 	switch (hci_pi(sk)->channel) {
14873a208627SMarcel Holtmann 	case HCI_CHANNEL_RAW:
14881da177e4SLinus Torvalds 		hci_sock_cmsg(sk, msg, skb);
14893a208627SMarcel Holtmann 		break;
149023500189SMarcel Holtmann 	case HCI_CHANNEL_USER:
1491cd82e61cSMarcel Holtmann 	case HCI_CHANNEL_MONITOR:
1492cd82e61cSMarcel Holtmann 		sock_recv_timestamp(msg, sk, skb);
1493cd82e61cSMarcel Holtmann 		break;
1494801c1e8dSJohan Hedberg 	default:
1495801c1e8dSJohan Hedberg 		if (hci_mgmt_chan_find(hci_pi(sk)->channel))
1496801c1e8dSJohan Hedberg 			sock_recv_timestamp(msg, sk, skb);
1497801c1e8dSJohan Hedberg 		break;
14983a208627SMarcel Holtmann 	}
14991da177e4SLinus Torvalds 
15001da177e4SLinus Torvalds 	skb_free_datagram(sk, skb);
15011da177e4SLinus Torvalds 
15024f34228bSLuiz Augusto von Dentz 	if (flags & MSG_TRUNC)
150383871f8cSDenis Kenzior 		copied = skblen;
150483871f8cSDenis Kenzior 
15051da177e4SLinus Torvalds 	return err ? : copied;
15061da177e4SLinus Torvalds }
15071da177e4SLinus Torvalds 
1508*0b59e272STetsuo Handa static int hci_mgmt_cmd(struct hci_mgmt_chan *chan, struct sock *sk, void *buf, size_t msglen)
1509fa4335d7SJohan Hedberg {
1510fa4335d7SJohan Hedberg 	u8 *cp;
1511fa4335d7SJohan Hedberg 	struct mgmt_hdr *hdr;
1512fa4335d7SJohan Hedberg 	u16 opcode, index, len;
1513fa4335d7SJohan Hedberg 	struct hci_dev *hdev = NULL;
1514fa4335d7SJohan Hedberg 	const struct hci_mgmt_handler *handler;
1515fa4335d7SJohan Hedberg 	bool var_len, no_hdev;
1516fa4335d7SJohan Hedberg 	int err;
1517fa4335d7SJohan Hedberg 
1518fa4335d7SJohan Hedberg 	BT_DBG("got %zu bytes", msglen);
1519fa4335d7SJohan Hedberg 
1520fa4335d7SJohan Hedberg 	if (msglen < sizeof(*hdr))
1521fa4335d7SJohan Hedberg 		return -EINVAL;
1522fa4335d7SJohan Hedberg 
1523fa4335d7SJohan Hedberg 	hdr = buf;
1524fa4335d7SJohan Hedberg 	opcode = __le16_to_cpu(hdr->opcode);
1525fa4335d7SJohan Hedberg 	index = __le16_to_cpu(hdr->index);
1526fa4335d7SJohan Hedberg 	len = __le16_to_cpu(hdr->len);
1527fa4335d7SJohan Hedberg 
1528fa4335d7SJohan Hedberg 	if (len != msglen - sizeof(*hdr)) {
1529fa4335d7SJohan Hedberg 		err = -EINVAL;
1530fa4335d7SJohan Hedberg 		goto done;
1531fa4335d7SJohan Hedberg 	}
1532fa4335d7SJohan Hedberg 
153338ceaa00SMarcel Holtmann 	if (chan->channel == HCI_CHANNEL_CONTROL) {
153438ceaa00SMarcel Holtmann 		struct sk_buff *skb;
153538ceaa00SMarcel Holtmann 
153638ceaa00SMarcel Holtmann 		/* Send event to monitor */
153738ceaa00SMarcel Holtmann 		skb = create_monitor_ctrl_command(sk, index, opcode, len,
153838ceaa00SMarcel Holtmann 						  buf + sizeof(*hdr));
153938ceaa00SMarcel Holtmann 		if (skb) {
154038ceaa00SMarcel Holtmann 			hci_send_to_channel(HCI_CHANNEL_MONITOR, skb,
154138ceaa00SMarcel Holtmann 					    HCI_SOCK_TRUSTED, NULL);
154238ceaa00SMarcel Holtmann 			kfree_skb(skb);
154338ceaa00SMarcel Holtmann 		}
154438ceaa00SMarcel Holtmann 	}
154538ceaa00SMarcel Holtmann 
1546fa4335d7SJohan Hedberg 	if (opcode >= chan->handler_count ||
1547fa4335d7SJohan Hedberg 	    chan->handlers[opcode].func == NULL) {
1548fa4335d7SJohan Hedberg 		BT_DBG("Unknown op %u", opcode);
1549fa4335d7SJohan Hedberg 		err = mgmt_cmd_status(sk, index, opcode,
1550fa4335d7SJohan Hedberg 				      MGMT_STATUS_UNKNOWN_COMMAND);
1551fa4335d7SJohan Hedberg 		goto done;
1552fa4335d7SJohan Hedberg 	}
1553fa4335d7SJohan Hedberg 
1554fa4335d7SJohan Hedberg 	handler = &chan->handlers[opcode];
1555fa4335d7SJohan Hedberg 
1556fa4335d7SJohan Hedberg 	if (!hci_sock_test_flag(sk, HCI_SOCK_TRUSTED) &&
1557fa4335d7SJohan Hedberg 	    !(handler->flags & HCI_MGMT_UNTRUSTED)) {
1558fa4335d7SJohan Hedberg 		err = mgmt_cmd_status(sk, index, opcode,
1559fa4335d7SJohan Hedberg 				      MGMT_STATUS_PERMISSION_DENIED);
1560fa4335d7SJohan Hedberg 		goto done;
1561fa4335d7SJohan Hedberg 	}
1562fa4335d7SJohan Hedberg 
1563fa4335d7SJohan Hedberg 	if (index != MGMT_INDEX_NONE) {
1564fa4335d7SJohan Hedberg 		hdev = hci_dev_get(index);
1565fa4335d7SJohan Hedberg 		if (!hdev) {
1566fa4335d7SJohan Hedberg 			err = mgmt_cmd_status(sk, index, opcode,
1567fa4335d7SJohan Hedberg 					      MGMT_STATUS_INVALID_INDEX);
1568fa4335d7SJohan Hedberg 			goto done;
1569fa4335d7SJohan Hedberg 		}
1570fa4335d7SJohan Hedberg 
1571fa4335d7SJohan Hedberg 		if (hci_dev_test_flag(hdev, HCI_SETUP) ||
1572fa4335d7SJohan Hedberg 		    hci_dev_test_flag(hdev, HCI_CONFIG) ||
1573fa4335d7SJohan Hedberg 		    hci_dev_test_flag(hdev, HCI_USER_CHANNEL)) {
1574fa4335d7SJohan Hedberg 			err = mgmt_cmd_status(sk, index, opcode,
1575fa4335d7SJohan Hedberg 					      MGMT_STATUS_INVALID_INDEX);
1576fa4335d7SJohan Hedberg 			goto done;
1577fa4335d7SJohan Hedberg 		}
1578fa4335d7SJohan Hedberg 
1579fa4335d7SJohan Hedberg 		if (hci_dev_test_flag(hdev, HCI_UNCONFIGURED) &&
1580fa4335d7SJohan Hedberg 		    !(handler->flags & HCI_MGMT_UNCONFIGURED)) {
1581fa4335d7SJohan Hedberg 			err = mgmt_cmd_status(sk, index, opcode,
1582fa4335d7SJohan Hedberg 					      MGMT_STATUS_INVALID_INDEX);
1583fa4335d7SJohan Hedberg 			goto done;
1584fa4335d7SJohan Hedberg 		}
1585fa4335d7SJohan Hedberg 	}
1586fa4335d7SJohan Hedberg 
1587d5cc6626SMarcel Holtmann 	if (!(handler->flags & HCI_MGMT_HDEV_OPTIONAL)) {
1588fa4335d7SJohan Hedberg 		no_hdev = (handler->flags & HCI_MGMT_NO_HDEV);
1589fa4335d7SJohan Hedberg 		if (no_hdev != !hdev) {
1590fa4335d7SJohan Hedberg 			err = mgmt_cmd_status(sk, index, opcode,
1591fa4335d7SJohan Hedberg 					      MGMT_STATUS_INVALID_INDEX);
1592fa4335d7SJohan Hedberg 			goto done;
1593fa4335d7SJohan Hedberg 		}
1594d5cc6626SMarcel Holtmann 	}
1595fa4335d7SJohan Hedberg 
1596fa4335d7SJohan Hedberg 	var_len = (handler->flags & HCI_MGMT_VAR_LEN);
1597fa4335d7SJohan Hedberg 	if ((var_len && len < handler->data_len) ||
1598fa4335d7SJohan Hedberg 	    (!var_len && len != handler->data_len)) {
1599fa4335d7SJohan Hedberg 		err = mgmt_cmd_status(sk, index, opcode,
1600fa4335d7SJohan Hedberg 				      MGMT_STATUS_INVALID_PARAMS);
1601fa4335d7SJohan Hedberg 		goto done;
1602fa4335d7SJohan Hedberg 	}
1603fa4335d7SJohan Hedberg 
1604fa4335d7SJohan Hedberg 	if (hdev && chan->hdev_init)
1605fa4335d7SJohan Hedberg 		chan->hdev_init(sk, hdev);
1606fa4335d7SJohan Hedberg 
1607fa4335d7SJohan Hedberg 	cp = buf + sizeof(*hdr);
1608fa4335d7SJohan Hedberg 
1609fa4335d7SJohan Hedberg 	err = handler->func(sk, hdev, cp, len);
1610fa4335d7SJohan Hedberg 	if (err < 0)
1611fa4335d7SJohan Hedberg 		goto done;
1612fa4335d7SJohan Hedberg 
1613fa4335d7SJohan Hedberg 	err = msglen;
1614fa4335d7SJohan Hedberg 
1615fa4335d7SJohan Hedberg done:
1616fa4335d7SJohan Hedberg 	if (hdev)
1617fa4335d7SJohan Hedberg 		hci_dev_put(hdev);
1618fa4335d7SJohan Hedberg 
1619fa4335d7SJohan Hedberg 	return err;
1620fa4335d7SJohan Hedberg }
1621fa4335d7SJohan Hedberg 
1622*0b59e272STetsuo Handa static int hci_logging_frame(struct sock *sk, void *buf, int len, unsigned int flags)
1623ac714949SMarcel Holtmann {
1624ac714949SMarcel Holtmann 	struct hci_mon_hdr *hdr;
1625ac714949SMarcel Holtmann 	struct sk_buff *skb;
1626ac714949SMarcel Holtmann 	struct hci_dev *hdev;
1627ac714949SMarcel Holtmann 	u16 index;
1628ac714949SMarcel Holtmann 	int err;
1629ac714949SMarcel Holtmann 
1630ac714949SMarcel Holtmann 	/* The logging frame consists at minimum of the standard header,
1631ac714949SMarcel Holtmann 	 * the priority byte, the ident length byte and at least one string
1632ac714949SMarcel Holtmann 	 * terminator NUL byte. Anything shorter are invalid packets.
1633ac714949SMarcel Holtmann 	 */
1634ac714949SMarcel Holtmann 	if (len < sizeof(*hdr) + 3)
1635ac714949SMarcel Holtmann 		return -EINVAL;
1636ac714949SMarcel Holtmann 
1637*0b59e272STetsuo Handa 	skb = bt_skb_send_alloc(sk, len, flags & MSG_DONTWAIT, &err);
1638ac714949SMarcel Holtmann 	if (!skb)
1639ac714949SMarcel Holtmann 		return err;
1640ac714949SMarcel Holtmann 
1641*0b59e272STetsuo Handa 	memcpy(skb_put(skb, len), buf, len);
1642ac714949SMarcel Holtmann 
1643ac714949SMarcel Holtmann 	hdr = (void *)skb->data;
1644ac714949SMarcel Holtmann 
1645ac714949SMarcel Holtmann 	if (__le16_to_cpu(hdr->len) != len - sizeof(*hdr)) {
1646ac714949SMarcel Holtmann 		err = -EINVAL;
1647ac714949SMarcel Holtmann 		goto drop;
1648ac714949SMarcel Holtmann 	}
1649ac714949SMarcel Holtmann 
1650ac714949SMarcel Holtmann 	if (__le16_to_cpu(hdr->opcode) == 0x0000) {
1651ac714949SMarcel Holtmann 		__u8 priority = skb->data[sizeof(*hdr)];
1652ac714949SMarcel Holtmann 		__u8 ident_len = skb->data[sizeof(*hdr) + 1];
1653ac714949SMarcel Holtmann 
1654ac714949SMarcel Holtmann 		/* Only the priorities 0-7 are valid and with that any other
1655ac714949SMarcel Holtmann 		 * value results in an invalid packet.
1656ac714949SMarcel Holtmann 		 *
1657ac714949SMarcel Holtmann 		 * The priority byte is followed by an ident length byte and
1658ac714949SMarcel Holtmann 		 * the NUL terminated ident string. Check that the ident
1659ac714949SMarcel Holtmann 		 * length is not overflowing the packet and also that the
1660ac714949SMarcel Holtmann 		 * ident string itself is NUL terminated. In case the ident
1661ac714949SMarcel Holtmann 		 * length is zero, the length value actually doubles as NUL
1662ac714949SMarcel Holtmann 		 * terminator identifier.
1663ac714949SMarcel Holtmann 		 *
1664ac714949SMarcel Holtmann 		 * The message follows the ident string (if present) and
1665ac714949SMarcel Holtmann 		 * must be NUL terminated. Otherwise it is not a valid packet.
1666ac714949SMarcel Holtmann 		 */
1667ac714949SMarcel Holtmann 		if (priority > 7 || skb->data[len - 1] != 0x00 ||
1668ac714949SMarcel Holtmann 		    ident_len > len - sizeof(*hdr) - 3 ||
1669ac714949SMarcel Holtmann 		    skb->data[sizeof(*hdr) + ident_len + 1] != 0x00) {
1670ac714949SMarcel Holtmann 			err = -EINVAL;
1671ac714949SMarcel Holtmann 			goto drop;
1672ac714949SMarcel Holtmann 		}
1673ac714949SMarcel Holtmann 	} else {
1674ac714949SMarcel Holtmann 		err = -EINVAL;
1675ac714949SMarcel Holtmann 		goto drop;
1676ac714949SMarcel Holtmann 	}
1677ac714949SMarcel Holtmann 
1678ac714949SMarcel Holtmann 	index = __le16_to_cpu(hdr->index);
1679ac714949SMarcel Holtmann 
1680ac714949SMarcel Holtmann 	if (index != MGMT_INDEX_NONE) {
1681ac714949SMarcel Holtmann 		hdev = hci_dev_get(index);
1682ac714949SMarcel Holtmann 		if (!hdev) {
1683ac714949SMarcel Holtmann 			err = -ENODEV;
1684ac714949SMarcel Holtmann 			goto drop;
1685ac714949SMarcel Holtmann 		}
1686ac714949SMarcel Holtmann 	} else {
1687ac714949SMarcel Holtmann 		hdev = NULL;
1688ac714949SMarcel Holtmann 	}
1689ac714949SMarcel Holtmann 
1690ac714949SMarcel Holtmann 	hdr->opcode = cpu_to_le16(HCI_MON_USER_LOGGING);
1691ac714949SMarcel Holtmann 
1692ac714949SMarcel Holtmann 	hci_send_to_channel(HCI_CHANNEL_MONITOR, skb, HCI_SOCK_TRUSTED, NULL);
1693ac714949SMarcel Holtmann 	err = len;
1694ac714949SMarcel Holtmann 
1695ac714949SMarcel Holtmann 	if (hdev)
1696ac714949SMarcel Holtmann 		hci_dev_put(hdev);
1697ac714949SMarcel Holtmann 
1698ac714949SMarcel Holtmann drop:
1699ac714949SMarcel Holtmann 	kfree_skb(skb);
1700ac714949SMarcel Holtmann 	return err;
1701ac714949SMarcel Holtmann }
1702ac714949SMarcel Holtmann 
17031b784140SYing Xue static int hci_sock_sendmsg(struct socket *sock, struct msghdr *msg,
17041b784140SYing Xue 			    size_t len)
17051da177e4SLinus Torvalds {
17061da177e4SLinus Torvalds 	struct sock *sk = sock->sk;
1707801c1e8dSJohan Hedberg 	struct hci_mgmt_chan *chan;
17081da177e4SLinus Torvalds 	struct hci_dev *hdev;
17091da177e4SLinus Torvalds 	struct sk_buff *skb;
17101da177e4SLinus Torvalds 	int err;
1711*0b59e272STetsuo Handa 	void *buf;
1712*0b59e272STetsuo Handa 	const unsigned int flags = msg->msg_flags;
17131da177e4SLinus Torvalds 
17141da177e4SLinus Torvalds 	BT_DBG("sock %p sk %p", sock, sk);
17151da177e4SLinus Torvalds 
1716*0b59e272STetsuo Handa 	if (flags & MSG_OOB)
17171da177e4SLinus Torvalds 		return -EOPNOTSUPP;
17181da177e4SLinus Torvalds 
1719*0b59e272STetsuo Handa 	if (flags & ~(MSG_DONTWAIT | MSG_NOSIGNAL | MSG_ERRQUEUE | MSG_CMSG_COMPAT))
17201da177e4SLinus Torvalds 		return -EINVAL;
17211da177e4SLinus Torvalds 
17221da177e4SLinus Torvalds 	if (len < 4 || len > HCI_MAX_FRAME_SIZE)
17231da177e4SLinus Torvalds 		return -EINVAL;
17241da177e4SLinus Torvalds 
1725*0b59e272STetsuo Handa 	buf = kmalloc(len, GFP_KERNEL);
1726*0b59e272STetsuo Handa 	if (!buf)
1727*0b59e272STetsuo Handa 		return -ENOMEM;
1728*0b59e272STetsuo Handa 	if (memcpy_from_msg(buf, msg, len)) {
1729*0b59e272STetsuo Handa 		kfree(buf);
1730*0b59e272STetsuo Handa 		return -EFAULT;
1731*0b59e272STetsuo Handa 	}
1732*0b59e272STetsuo Handa 
17331da177e4SLinus Torvalds 	lock_sock(sk);
17341da177e4SLinus Torvalds 
17350381101fSJohan Hedberg 	switch (hci_pi(sk)->channel) {
17360381101fSJohan Hedberg 	case HCI_CHANNEL_RAW:
173723500189SMarcel Holtmann 	case HCI_CHANNEL_USER:
17380381101fSJohan Hedberg 		break;
1739cd82e61cSMarcel Holtmann 	case HCI_CHANNEL_MONITOR:
1740cd82e61cSMarcel Holtmann 		err = -EOPNOTSUPP;
1741cd82e61cSMarcel Holtmann 		goto done;
1742ac714949SMarcel Holtmann 	case HCI_CHANNEL_LOGGING:
1743*0b59e272STetsuo Handa 		err = hci_logging_frame(sk, buf, len, flags);
1744ac714949SMarcel Holtmann 		goto done;
17450381101fSJohan Hedberg 	default:
1746801c1e8dSJohan Hedberg 		mutex_lock(&mgmt_chan_list_lock);
1747801c1e8dSJohan Hedberg 		chan = __hci_mgmt_chan_find(hci_pi(sk)->channel);
1748801c1e8dSJohan Hedberg 		if (chan)
1749*0b59e272STetsuo Handa 			err = hci_mgmt_cmd(chan, sk, buf, len);
1750801c1e8dSJohan Hedberg 		else
17510381101fSJohan Hedberg 			err = -EINVAL;
1752801c1e8dSJohan Hedberg 
1753801c1e8dSJohan Hedberg 		mutex_unlock(&mgmt_chan_list_lock);
17540381101fSJohan Hedberg 		goto done;
17550381101fSJohan Hedberg 	}
17560381101fSJohan Hedberg 
1757e0448092STetsuo Handa 	hdev = hci_hdev_from_sock(sk);
1758e0448092STetsuo Handa 	if (IS_ERR(hdev)) {
1759e0448092STetsuo Handa 		err = PTR_ERR(hdev);
17601da177e4SLinus Torvalds 		goto done;
17611da177e4SLinus Torvalds 	}
17621da177e4SLinus Torvalds 
17637e21addcSMarcel Holtmann 	if (!test_bit(HCI_UP, &hdev->flags)) {
17647e21addcSMarcel Holtmann 		err = -ENETDOWN;
17657e21addcSMarcel Holtmann 		goto done;
17667e21addcSMarcel Holtmann 	}
17677e21addcSMarcel Holtmann 
1768*0b59e272STetsuo Handa 	skb = bt_skb_send_alloc(sk, len, flags & MSG_DONTWAIT, &err);
176970f23020SAndrei Emeltchenko 	if (!skb)
17701da177e4SLinus Torvalds 		goto done;
17711da177e4SLinus Torvalds 
1772*0b59e272STetsuo Handa 	memcpy(skb_put(skb, len), buf, len);
17731da177e4SLinus Torvalds 
17748528d3f7SMarcel Holtmann 	hci_skb_pkt_type(skb) = skb->data[0];
17751da177e4SLinus Torvalds 	skb_pull(skb, 1);
17761da177e4SLinus Torvalds 
17771bc5ad16SMarcel Holtmann 	if (hci_pi(sk)->channel == HCI_CHANNEL_USER) {
17781bc5ad16SMarcel Holtmann 		/* No permission check is needed for user channel
17791bc5ad16SMarcel Holtmann 		 * since that gets enforced when binding the socket.
17801bc5ad16SMarcel Holtmann 		 *
17811bc5ad16SMarcel Holtmann 		 * However check that the packet type is valid.
17821bc5ad16SMarcel Holtmann 		 */
1783d79f34e3SMarcel Holtmann 		if (hci_skb_pkt_type(skb) != HCI_COMMAND_PKT &&
1784d79f34e3SMarcel Holtmann 		    hci_skb_pkt_type(skb) != HCI_ACLDATA_PKT &&
1785cc974003SMarcel Holtmann 		    hci_skb_pkt_type(skb) != HCI_SCODATA_PKT &&
1786cc974003SMarcel Holtmann 		    hci_skb_pkt_type(skb) != HCI_ISODATA_PKT) {
17871bc5ad16SMarcel Holtmann 			err = -EINVAL;
17881bc5ad16SMarcel Holtmann 			goto drop;
17891bc5ad16SMarcel Holtmann 		}
17901bc5ad16SMarcel Holtmann 
17911bc5ad16SMarcel Holtmann 		skb_queue_tail(&hdev->raw_q, skb);
17921bc5ad16SMarcel Holtmann 		queue_work(hdev->workqueue, &hdev->tx_work);
1793d79f34e3SMarcel Holtmann 	} else if (hci_skb_pkt_type(skb) == HCI_COMMAND_PKT) {
179483985319SHarvey Harrison 		u16 opcode = get_unaligned_le16(skb->data);
17951da177e4SLinus Torvalds 		u16 ogf = hci_opcode_ogf(opcode);
17961da177e4SLinus Torvalds 		u16 ocf = hci_opcode_ocf(opcode);
17971da177e4SLinus Torvalds 
17981da177e4SLinus Torvalds 		if (((ogf > HCI_SFLT_MAX_OGF) ||
17993bb3c755SGustavo Padovan 		     !hci_test_bit(ocf & HCI_FLT_OCF_BITS,
18003bb3c755SGustavo Padovan 				   &hci_sec_filter.ocf_mask[ogf])) &&
18011da177e4SLinus Torvalds 		    !capable(CAP_NET_RAW)) {
18021da177e4SLinus Torvalds 			err = -EPERM;
18031da177e4SLinus Torvalds 			goto drop;
18041da177e4SLinus Torvalds 		}
18051da177e4SLinus Torvalds 
18061982162bSMarcel Holtmann 		/* Since the opcode has already been extracted here, store
18071982162bSMarcel Holtmann 		 * a copy of the value for later use by the drivers.
18081982162bSMarcel Holtmann 		 */
18091982162bSMarcel Holtmann 		hci_skb_opcode(skb) = opcode;
18101982162bSMarcel Holtmann 
1811fee746b0SMarcel Holtmann 		if (ogf == 0x3f) {
18121da177e4SLinus Torvalds 			skb_queue_tail(&hdev->raw_q, skb);
18133eff45eaSGustavo F. Padovan 			queue_work(hdev->workqueue, &hdev->tx_work);
18141da177e4SLinus Torvalds 		} else {
181549c922bbSStephen Hemminger 			/* Stand-alone HCI commands must be flagged as
181611714b3dSJohan Hedberg 			 * single-command requests.
181711714b3dSJohan Hedberg 			 */
181844d27137SJohan Hedberg 			bt_cb(skb)->hci.req_flags |= HCI_REQ_START;
181911714b3dSJohan Hedberg 
18201da177e4SLinus Torvalds 			skb_queue_tail(&hdev->cmd_q, skb);
1821c347b765SGustavo F. Padovan 			queue_work(hdev->workqueue, &hdev->cmd_work);
18221da177e4SLinus Torvalds 		}
18231da177e4SLinus Torvalds 	} else {
18241da177e4SLinus Torvalds 		if (!capable(CAP_NET_RAW)) {
18251da177e4SLinus Torvalds 			err = -EPERM;
18261da177e4SLinus Torvalds 			goto drop;
18271da177e4SLinus Torvalds 		}
18281da177e4SLinus Torvalds 
1829d79f34e3SMarcel Holtmann 		if (hci_skb_pkt_type(skb) != HCI_ACLDATA_PKT &&
1830cc974003SMarcel Holtmann 		    hci_skb_pkt_type(skb) != HCI_SCODATA_PKT &&
1831cc974003SMarcel Holtmann 		    hci_skb_pkt_type(skb) != HCI_ISODATA_PKT) {
1832bb77543eSMarcel Holtmann 			err = -EINVAL;
1833bb77543eSMarcel Holtmann 			goto drop;
1834bb77543eSMarcel Holtmann 		}
1835bb77543eSMarcel Holtmann 
18361da177e4SLinus Torvalds 		skb_queue_tail(&hdev->raw_q, skb);
18373eff45eaSGustavo F. Padovan 		queue_work(hdev->workqueue, &hdev->tx_work);
18381da177e4SLinus Torvalds 	}
18391da177e4SLinus Torvalds 
18401da177e4SLinus Torvalds 	err = len;
18411da177e4SLinus Torvalds 
18421da177e4SLinus Torvalds done:
18431da177e4SLinus Torvalds 	release_sock(sk);
1844*0b59e272STetsuo Handa 	kfree(buf);
18451da177e4SLinus Torvalds 	return err;
18461da177e4SLinus Torvalds 
18471da177e4SLinus Torvalds drop:
18481da177e4SLinus Torvalds 	kfree_skb(skb);
18491da177e4SLinus Torvalds 	goto done;
18501da177e4SLinus Torvalds }
18511da177e4SLinus Torvalds 
18528fc9ced3SGustavo Padovan static int hci_sock_setsockopt(struct socket *sock, int level, int optname,
1853a7b75c5aSChristoph Hellwig 			       sockptr_t optval, unsigned int len)
18541da177e4SLinus Torvalds {
18551da177e4SLinus Torvalds 	struct hci_ufilter uf = { .opcode = 0 };
18561da177e4SLinus Torvalds 	struct sock *sk = sock->sk;
18571da177e4SLinus Torvalds 	int err = 0, opt = 0;
18581da177e4SLinus Torvalds 
18591da177e4SLinus Torvalds 	BT_DBG("sk %p, opt %d", sk, optname);
18601da177e4SLinus Torvalds 
186147b0f573SMarcel Holtmann 	if (level != SOL_HCI)
186247b0f573SMarcel Holtmann 		return -ENOPROTOOPT;
186347b0f573SMarcel Holtmann 
18641da177e4SLinus Torvalds 	lock_sock(sk);
18651da177e4SLinus Torvalds 
18662f39cdb7SMarcel Holtmann 	if (hci_pi(sk)->channel != HCI_CHANNEL_RAW) {
1867c2371e80SMarcel Holtmann 		err = -EBADFD;
18682f39cdb7SMarcel Holtmann 		goto done;
18692f39cdb7SMarcel Holtmann 	}
18702f39cdb7SMarcel Holtmann 
18711da177e4SLinus Torvalds 	switch (optname) {
18721da177e4SLinus Torvalds 	case HCI_DATA_DIR:
1873a7b75c5aSChristoph Hellwig 		if (copy_from_sockptr(&opt, optval, sizeof(opt))) {
18741da177e4SLinus Torvalds 			err = -EFAULT;
18751da177e4SLinus Torvalds 			break;
18761da177e4SLinus Torvalds 		}
18771da177e4SLinus Torvalds 
18781da177e4SLinus Torvalds 		if (opt)
18791da177e4SLinus Torvalds 			hci_pi(sk)->cmsg_mask |= HCI_CMSG_DIR;
18801da177e4SLinus Torvalds 		else
18811da177e4SLinus Torvalds 			hci_pi(sk)->cmsg_mask &= ~HCI_CMSG_DIR;
18821da177e4SLinus Torvalds 		break;
18831da177e4SLinus Torvalds 
18841da177e4SLinus Torvalds 	case HCI_TIME_STAMP:
1885a7b75c5aSChristoph Hellwig 		if (copy_from_sockptr(&opt, optval, sizeof(opt))) {
18861da177e4SLinus Torvalds 			err = -EFAULT;
18871da177e4SLinus Torvalds 			break;
18881da177e4SLinus Torvalds 		}
18891da177e4SLinus Torvalds 
18901da177e4SLinus Torvalds 		if (opt)
18911da177e4SLinus Torvalds 			hci_pi(sk)->cmsg_mask |= HCI_CMSG_TSTAMP;
18921da177e4SLinus Torvalds 		else
18931da177e4SLinus Torvalds 			hci_pi(sk)->cmsg_mask &= ~HCI_CMSG_TSTAMP;
18941da177e4SLinus Torvalds 		break;
18951da177e4SLinus Torvalds 
18961da177e4SLinus Torvalds 	case HCI_FILTER:
18970878b666SMarcel Holtmann 		{
18980878b666SMarcel Holtmann 			struct hci_filter *f = &hci_pi(sk)->filter;
18990878b666SMarcel Holtmann 
19000878b666SMarcel Holtmann 			uf.type_mask = f->type_mask;
19010878b666SMarcel Holtmann 			uf.opcode    = f->opcode;
19020878b666SMarcel Holtmann 			uf.event_mask[0] = *((u32 *) f->event_mask + 0);
19030878b666SMarcel Holtmann 			uf.event_mask[1] = *((u32 *) f->event_mask + 1);
19040878b666SMarcel Holtmann 		}
19050878b666SMarcel Holtmann 
19061da177e4SLinus Torvalds 		len = min_t(unsigned int, len, sizeof(uf));
1907a7b75c5aSChristoph Hellwig 		if (copy_from_sockptr(&uf, optval, len)) {
19081da177e4SLinus Torvalds 			err = -EFAULT;
19091da177e4SLinus Torvalds 			break;
19101da177e4SLinus Torvalds 		}
19111da177e4SLinus Torvalds 
19121da177e4SLinus Torvalds 		if (!capable(CAP_NET_RAW)) {
19131da177e4SLinus Torvalds 			uf.type_mask &= hci_sec_filter.type_mask;
19141da177e4SLinus Torvalds 			uf.event_mask[0] &= *((u32 *) hci_sec_filter.event_mask + 0);
19151da177e4SLinus Torvalds 			uf.event_mask[1] &= *((u32 *) hci_sec_filter.event_mask + 1);
19161da177e4SLinus Torvalds 		}
19171da177e4SLinus Torvalds 
19181da177e4SLinus Torvalds 		{
19191da177e4SLinus Torvalds 			struct hci_filter *f = &hci_pi(sk)->filter;
19201da177e4SLinus Torvalds 
19211da177e4SLinus Torvalds 			f->type_mask = uf.type_mask;
19221da177e4SLinus Torvalds 			f->opcode    = uf.opcode;
19231da177e4SLinus Torvalds 			*((u32 *) f->event_mask + 0) = uf.event_mask[0];
19241da177e4SLinus Torvalds 			*((u32 *) f->event_mask + 1) = uf.event_mask[1];
19251da177e4SLinus Torvalds 		}
19261da177e4SLinus Torvalds 		break;
19271da177e4SLinus Torvalds 
19281da177e4SLinus Torvalds 	default:
19291da177e4SLinus Torvalds 		err = -ENOPROTOOPT;
19301da177e4SLinus Torvalds 		break;
19311da177e4SLinus Torvalds 	}
19321da177e4SLinus Torvalds 
19332f39cdb7SMarcel Holtmann done:
19341da177e4SLinus Torvalds 	release_sock(sk);
19351da177e4SLinus Torvalds 	return err;
19361da177e4SLinus Torvalds }
19371da177e4SLinus Torvalds 
19388fc9ced3SGustavo Padovan static int hci_sock_getsockopt(struct socket *sock, int level, int optname,
19398fc9ced3SGustavo Padovan 			       char __user *optval, int __user *optlen)
19401da177e4SLinus Torvalds {
19411da177e4SLinus Torvalds 	struct hci_ufilter uf;
19421da177e4SLinus Torvalds 	struct sock *sk = sock->sk;
1943cedc5469SMarcel Holtmann 	int len, opt, err = 0;
1944cedc5469SMarcel Holtmann 
1945cedc5469SMarcel Holtmann 	BT_DBG("sk %p, opt %d", sk, optname);
19461da177e4SLinus Torvalds 
194747b0f573SMarcel Holtmann 	if (level != SOL_HCI)
194847b0f573SMarcel Holtmann 		return -ENOPROTOOPT;
194947b0f573SMarcel Holtmann 
19501da177e4SLinus Torvalds 	if (get_user(len, optlen))
19511da177e4SLinus Torvalds 		return -EFAULT;
19521da177e4SLinus Torvalds 
1953cedc5469SMarcel Holtmann 	lock_sock(sk);
1954cedc5469SMarcel Holtmann 
1955cedc5469SMarcel Holtmann 	if (hci_pi(sk)->channel != HCI_CHANNEL_RAW) {
1956c2371e80SMarcel Holtmann 		err = -EBADFD;
1957cedc5469SMarcel Holtmann 		goto done;
1958cedc5469SMarcel Holtmann 	}
1959cedc5469SMarcel Holtmann 
19601da177e4SLinus Torvalds 	switch (optname) {
19611da177e4SLinus Torvalds 	case HCI_DATA_DIR:
19621da177e4SLinus Torvalds 		if (hci_pi(sk)->cmsg_mask & HCI_CMSG_DIR)
19631da177e4SLinus Torvalds 			opt = 1;
19641da177e4SLinus Torvalds 		else
19651da177e4SLinus Torvalds 			opt = 0;
19661da177e4SLinus Torvalds 
19671da177e4SLinus Torvalds 		if (put_user(opt, optval))
1968cedc5469SMarcel Holtmann 			err = -EFAULT;
19691da177e4SLinus Torvalds 		break;
19701da177e4SLinus Torvalds 
19711da177e4SLinus Torvalds 	case HCI_TIME_STAMP:
19721da177e4SLinus Torvalds 		if (hci_pi(sk)->cmsg_mask & HCI_CMSG_TSTAMP)
19731da177e4SLinus Torvalds 			opt = 1;
19741da177e4SLinus Torvalds 		else
19751da177e4SLinus Torvalds 			opt = 0;
19761da177e4SLinus Torvalds 
19771da177e4SLinus Torvalds 		if (put_user(opt, optval))
1978cedc5469SMarcel Holtmann 			err = -EFAULT;
19791da177e4SLinus Torvalds 		break;
19801da177e4SLinus Torvalds 
19811da177e4SLinus Torvalds 	case HCI_FILTER:
19821da177e4SLinus Torvalds 		{
19831da177e4SLinus Torvalds 			struct hci_filter *f = &hci_pi(sk)->filter;
19841da177e4SLinus Torvalds 
1985e15ca9a0SMathias Krause 			memset(&uf, 0, sizeof(uf));
19861da177e4SLinus Torvalds 			uf.type_mask = f->type_mask;
19871da177e4SLinus Torvalds 			uf.opcode    = f->opcode;
19881da177e4SLinus Torvalds 			uf.event_mask[0] = *((u32 *) f->event_mask + 0);
19891da177e4SLinus Torvalds 			uf.event_mask[1] = *((u32 *) f->event_mask + 1);
19901da177e4SLinus Torvalds 		}
19911da177e4SLinus Torvalds 
19921da177e4SLinus Torvalds 		len = min_t(unsigned int, len, sizeof(uf));
19931da177e4SLinus Torvalds 		if (copy_to_user(optval, &uf, len))
1994cedc5469SMarcel Holtmann 			err = -EFAULT;
19951da177e4SLinus Torvalds 		break;
19961da177e4SLinus Torvalds 
19971da177e4SLinus Torvalds 	default:
1998cedc5469SMarcel Holtmann 		err = -ENOPROTOOPT;
19991da177e4SLinus Torvalds 		break;
20001da177e4SLinus Torvalds 	}
20011da177e4SLinus Torvalds 
2002cedc5469SMarcel Holtmann done:
2003cedc5469SMarcel Holtmann 	release_sock(sk);
2004cedc5469SMarcel Holtmann 	return err;
20051da177e4SLinus Torvalds }
20061da177e4SLinus Torvalds 
200790ddc4f0SEric Dumazet static const struct proto_ops hci_sock_ops = {
20081da177e4SLinus Torvalds 	.family		= PF_BLUETOOTH,
20091da177e4SLinus Torvalds 	.owner		= THIS_MODULE,
20101da177e4SLinus Torvalds 	.release	= hci_sock_release,
20111da177e4SLinus Torvalds 	.bind		= hci_sock_bind,
20121da177e4SLinus Torvalds 	.getname	= hci_sock_getname,
20131da177e4SLinus Torvalds 	.sendmsg	= hci_sock_sendmsg,
20141da177e4SLinus Torvalds 	.recvmsg	= hci_sock_recvmsg,
20151da177e4SLinus Torvalds 	.ioctl		= hci_sock_ioctl,
20167a6038b3SArnd Bergmann #ifdef CONFIG_COMPAT
20177a6038b3SArnd Bergmann 	.compat_ioctl	= hci_sock_compat_ioctl,
20187a6038b3SArnd Bergmann #endif
2019a11e1d43SLinus Torvalds 	.poll		= datagram_poll,
20201da177e4SLinus Torvalds 	.listen		= sock_no_listen,
20211da177e4SLinus Torvalds 	.shutdown	= sock_no_shutdown,
20221da177e4SLinus Torvalds 	.setsockopt	= hci_sock_setsockopt,
20231da177e4SLinus Torvalds 	.getsockopt	= hci_sock_getsockopt,
20241da177e4SLinus Torvalds 	.connect	= sock_no_connect,
20251da177e4SLinus Torvalds 	.socketpair	= sock_no_socketpair,
20261da177e4SLinus Torvalds 	.accept		= sock_no_accept,
20271da177e4SLinus Torvalds 	.mmap		= sock_no_mmap
20281da177e4SLinus Torvalds };
20291da177e4SLinus Torvalds 
20301da177e4SLinus Torvalds static struct proto hci_sk_proto = {
20311da177e4SLinus Torvalds 	.name		= "HCI",
20321da177e4SLinus Torvalds 	.owner		= THIS_MODULE,
20331da177e4SLinus Torvalds 	.obj_size	= sizeof(struct hci_pinfo)
20341da177e4SLinus Torvalds };
20351da177e4SLinus Torvalds 
20363f378b68SEric Paris static int hci_sock_create(struct net *net, struct socket *sock, int protocol,
20373f378b68SEric Paris 			   int kern)
20381da177e4SLinus Torvalds {
20391da177e4SLinus Torvalds 	struct sock *sk;
20401da177e4SLinus Torvalds 
20411da177e4SLinus Torvalds 	BT_DBG("sock %p", sock);
20421da177e4SLinus Torvalds 
20431da177e4SLinus Torvalds 	if (sock->type != SOCK_RAW)
20441da177e4SLinus Torvalds 		return -ESOCKTNOSUPPORT;
20451da177e4SLinus Torvalds 
20461da177e4SLinus Torvalds 	sock->ops = &hci_sock_ops;
20471da177e4SLinus Torvalds 
204811aa9c28SEric W. Biederman 	sk = sk_alloc(net, PF_BLUETOOTH, GFP_ATOMIC, &hci_sk_proto, kern);
20491da177e4SLinus Torvalds 	if (!sk)
20501da177e4SLinus Torvalds 		return -ENOMEM;
20511da177e4SLinus Torvalds 
20521da177e4SLinus Torvalds 	sock_init_data(sock, sk);
20531da177e4SLinus Torvalds 
20541da177e4SLinus Torvalds 	sock_reset_flag(sk, SOCK_ZAPPED);
20551da177e4SLinus Torvalds 
20561da177e4SLinus Torvalds 	sk->sk_protocol = protocol;
20571da177e4SLinus Torvalds 
20581da177e4SLinus Torvalds 	sock->state = SS_UNCONNECTED;
20591da177e4SLinus Torvalds 	sk->sk_state = BT_OPEN;
20601da177e4SLinus Torvalds 
20611da177e4SLinus Torvalds 	bt_sock_link(&hci_sk_list, sk);
20621da177e4SLinus Torvalds 	return 0;
20631da177e4SLinus Torvalds }
20641da177e4SLinus Torvalds 
2065ec1b4cf7SStephen Hemminger static const struct net_proto_family hci_sock_family_ops = {
20661da177e4SLinus Torvalds 	.family	= PF_BLUETOOTH,
20671da177e4SLinus Torvalds 	.owner	= THIS_MODULE,
20681da177e4SLinus Torvalds 	.create	= hci_sock_create,
20691da177e4SLinus Torvalds };
20701da177e4SLinus Torvalds 
20711da177e4SLinus Torvalds int __init hci_sock_init(void)
20721da177e4SLinus Torvalds {
20731da177e4SLinus Torvalds 	int err;
20741da177e4SLinus Torvalds 
2075b0a8e282SMarcel Holtmann 	BUILD_BUG_ON(sizeof(struct sockaddr_hci) > sizeof(struct sockaddr));
2076b0a8e282SMarcel Holtmann 
20771da177e4SLinus Torvalds 	err = proto_register(&hci_sk_proto, 0);
20781da177e4SLinus Torvalds 	if (err < 0)
20791da177e4SLinus Torvalds 		return err;
20801da177e4SLinus Torvalds 
20811da177e4SLinus Torvalds 	err = bt_sock_register(BTPROTO_HCI, &hci_sock_family_ops);
2082f7c86637SMasatake YAMATO 	if (err < 0) {
2083f7c86637SMasatake YAMATO 		BT_ERR("HCI socket registration failed");
20841da177e4SLinus Torvalds 		goto error;
2085f7c86637SMasatake YAMATO 	}
2086f7c86637SMasatake YAMATO 
2087b0316615SAl Viro 	err = bt_procfs_init(&init_net, "hci", &hci_sk_list, NULL);
2088f7c86637SMasatake YAMATO 	if (err < 0) {
2089f7c86637SMasatake YAMATO 		BT_ERR("Failed to create HCI proc file");
2090f7c86637SMasatake YAMATO 		bt_sock_unregister(BTPROTO_HCI);
2091f7c86637SMasatake YAMATO 		goto error;
2092f7c86637SMasatake YAMATO 	}
20931da177e4SLinus Torvalds 
20941da177e4SLinus Torvalds 	BT_INFO("HCI socket layer initialized");
20951da177e4SLinus Torvalds 
20961da177e4SLinus Torvalds 	return 0;
20971da177e4SLinus Torvalds 
20981da177e4SLinus Torvalds error:
20991da177e4SLinus Torvalds 	proto_unregister(&hci_sk_proto);
21001da177e4SLinus Torvalds 	return err;
21011da177e4SLinus Torvalds }
21021da177e4SLinus Torvalds 
2103b7440a14SAnand Gadiyar void hci_sock_cleanup(void)
21041da177e4SLinus Torvalds {
2105f7c86637SMasatake YAMATO 	bt_procfs_cleanup(&init_net, "hci");
21065e9d7f86SDavid Herrmann 	bt_sock_unregister(BTPROTO_HCI);
21071da177e4SLinus Torvalds 	proto_unregister(&hci_sk_proto);
21081da177e4SLinus Torvalds }
2109