xref: /openbmc/linux/net/bluetooth/hci_sock.c (revision 7a6038b30017c45e1110388083849689356a23ae)
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. */
26*7a6038b3SArnd Bergmann #include <linux/compat.h>
278c520a59SGustavo Padovan #include <linux/export.h>
28787b306cSJohannes Berg #include <linux/utsname.h>
2970ecce91SMarcel Holtmann #include <linux/sched.h>
301da177e4SLinus Torvalds #include <asm/unaligned.h>
311da177e4SLinus Torvalds 
321da177e4SLinus Torvalds #include <net/bluetooth/bluetooth.h>
331da177e4SLinus Torvalds #include <net/bluetooth/hci_core.h>
34cd82e61cSMarcel Holtmann #include <net/bluetooth/hci_mon.h>
35fa4335d7SJohan Hedberg #include <net/bluetooth/mgmt.h>
36fa4335d7SJohan Hedberg 
37fa4335d7SJohan Hedberg #include "mgmt_util.h"
381da177e4SLinus Torvalds 
39801c1e8dSJohan Hedberg static LIST_HEAD(mgmt_chan_list);
40801c1e8dSJohan Hedberg static DEFINE_MUTEX(mgmt_chan_list_lock);
41801c1e8dSJohan Hedberg 
4270ecce91SMarcel Holtmann static DEFINE_IDA(sock_cookie_ida);
4370ecce91SMarcel Holtmann 
44cd82e61cSMarcel Holtmann static atomic_t monitor_promisc = ATOMIC_INIT(0);
45cd82e61cSMarcel Holtmann 
461da177e4SLinus Torvalds /* ----- HCI socket interface ----- */
471da177e4SLinus Torvalds 
48863def58SMarcel Holtmann /* Socket info */
49863def58SMarcel Holtmann #define hci_pi(sk) ((struct hci_pinfo *) sk)
50863def58SMarcel Holtmann 
51863def58SMarcel Holtmann struct hci_pinfo {
52863def58SMarcel Holtmann 	struct bt_sock    bt;
53863def58SMarcel Holtmann 	struct hci_dev    *hdev;
54863def58SMarcel Holtmann 	struct hci_filter filter;
55863def58SMarcel Holtmann 	__u32             cmsg_mask;
56863def58SMarcel Holtmann 	unsigned short    channel;
576befc644SMarcel Holtmann 	unsigned long     flags;
5870ecce91SMarcel Holtmann 	__u32             cookie;
5970ecce91SMarcel Holtmann 	char              comm[TASK_COMM_LEN];
60863def58SMarcel Holtmann };
61863def58SMarcel Holtmann 
626befc644SMarcel Holtmann void hci_sock_set_flag(struct sock *sk, int nr)
636befc644SMarcel Holtmann {
646befc644SMarcel Holtmann 	set_bit(nr, &hci_pi(sk)->flags);
656befc644SMarcel Holtmann }
666befc644SMarcel Holtmann 
676befc644SMarcel Holtmann void hci_sock_clear_flag(struct sock *sk, int nr)
686befc644SMarcel Holtmann {
696befc644SMarcel Holtmann 	clear_bit(nr, &hci_pi(sk)->flags);
706befc644SMarcel Holtmann }
716befc644SMarcel Holtmann 
72c85be545SMarcel Holtmann int hci_sock_test_flag(struct sock *sk, int nr)
73c85be545SMarcel Holtmann {
74c85be545SMarcel Holtmann 	return test_bit(nr, &hci_pi(sk)->flags);
75c85be545SMarcel Holtmann }
76c85be545SMarcel Holtmann 
77d0f172b1SJohan Hedberg unsigned short hci_sock_get_channel(struct sock *sk)
78d0f172b1SJohan Hedberg {
79d0f172b1SJohan Hedberg 	return hci_pi(sk)->channel;
80d0f172b1SJohan Hedberg }
81d0f172b1SJohan Hedberg 
8270ecce91SMarcel Holtmann u32 hci_sock_get_cookie(struct sock *sk)
8370ecce91SMarcel Holtmann {
8470ecce91SMarcel Holtmann 	return hci_pi(sk)->cookie;
8570ecce91SMarcel Holtmann }
8670ecce91SMarcel Holtmann 
87df1cb87aSMarcel Holtmann static bool hci_sock_gen_cookie(struct sock *sk)
88df1cb87aSMarcel Holtmann {
89df1cb87aSMarcel Holtmann 	int id = hci_pi(sk)->cookie;
90df1cb87aSMarcel Holtmann 
91df1cb87aSMarcel Holtmann 	if (!id) {
92df1cb87aSMarcel Holtmann 		id = ida_simple_get(&sock_cookie_ida, 1, 0, GFP_KERNEL);
93df1cb87aSMarcel Holtmann 		if (id < 0)
94df1cb87aSMarcel Holtmann 			id = 0xffffffff;
95df1cb87aSMarcel Holtmann 
96df1cb87aSMarcel Holtmann 		hci_pi(sk)->cookie = id;
97df1cb87aSMarcel Holtmann 		get_task_comm(hci_pi(sk)->comm, current);
98df1cb87aSMarcel Holtmann 		return true;
99df1cb87aSMarcel Holtmann 	}
100df1cb87aSMarcel Holtmann 
101df1cb87aSMarcel Holtmann 	return false;
102df1cb87aSMarcel Holtmann }
103df1cb87aSMarcel Holtmann 
104df1cb87aSMarcel Holtmann static void hci_sock_free_cookie(struct sock *sk)
105df1cb87aSMarcel Holtmann {
106df1cb87aSMarcel Holtmann 	int id = hci_pi(sk)->cookie;
107df1cb87aSMarcel Holtmann 
108df1cb87aSMarcel Holtmann 	if (id) {
109df1cb87aSMarcel Holtmann 		hci_pi(sk)->cookie = 0xffffffff;
110df1cb87aSMarcel Holtmann 		ida_simple_remove(&sock_cookie_ida, id);
111df1cb87aSMarcel Holtmann 	}
112df1cb87aSMarcel Holtmann }
113df1cb87aSMarcel Holtmann 
1149391976aSJiri Slaby static inline int hci_test_bit(int nr, const void *addr)
1151da177e4SLinus Torvalds {
1169391976aSJiri Slaby 	return *((const __u32 *) addr + (nr >> 5)) & ((__u32) 1 << (nr & 31));
1171da177e4SLinus Torvalds }
1181da177e4SLinus Torvalds 
1191da177e4SLinus Torvalds /* Security filter */
1203ad254f7SMarcel Holtmann #define HCI_SFLT_MAX_OGF  5
1213ad254f7SMarcel Holtmann 
1223ad254f7SMarcel Holtmann struct hci_sec_filter {
1233ad254f7SMarcel Holtmann 	__u32 type_mask;
1243ad254f7SMarcel Holtmann 	__u32 event_mask[2];
1253ad254f7SMarcel Holtmann 	__u32 ocf_mask[HCI_SFLT_MAX_OGF + 1][4];
1263ad254f7SMarcel Holtmann };
1273ad254f7SMarcel Holtmann 
1287e67c112SMarcel Holtmann static const struct hci_sec_filter hci_sec_filter = {
1291da177e4SLinus Torvalds 	/* Packet types */
1301da177e4SLinus Torvalds 	0x10,
1311da177e4SLinus Torvalds 	/* Events */
132dd7f5527SMarcel Holtmann 	{ 0x1000d9fe, 0x0000b00c },
1331da177e4SLinus Torvalds 	/* Commands */
1341da177e4SLinus Torvalds 	{
1351da177e4SLinus Torvalds 		{ 0x0 },
1361da177e4SLinus Torvalds 		/* OGF_LINK_CTL */
1377c631a67SMarcel Holtmann 		{ 0xbe000006, 0x00000001, 0x00000000, 0x00 },
1381da177e4SLinus Torvalds 		/* OGF_LINK_POLICY */
1397c631a67SMarcel Holtmann 		{ 0x00005200, 0x00000000, 0x00000000, 0x00 },
1401da177e4SLinus Torvalds 		/* OGF_HOST_CTL */
1417c631a67SMarcel Holtmann 		{ 0xaab00200, 0x2b402aaa, 0x05220154, 0x00 },
1421da177e4SLinus Torvalds 		/* OGF_INFO_PARAM */
1437c631a67SMarcel Holtmann 		{ 0x000002be, 0x00000000, 0x00000000, 0x00 },
1441da177e4SLinus Torvalds 		/* OGF_STATUS_PARAM */
1457c631a67SMarcel Holtmann 		{ 0x000000ea, 0x00000000, 0x00000000, 0x00 }
1461da177e4SLinus Torvalds 	}
1471da177e4SLinus Torvalds };
1481da177e4SLinus Torvalds 
1491da177e4SLinus Torvalds static struct bt_sock_list hci_sk_list = {
150d5fb2962SRobert P. J. Day 	.lock = __RW_LOCK_UNLOCKED(hci_sk_list.lock)
1511da177e4SLinus Torvalds };
1521da177e4SLinus Torvalds 
153f81fe64fSMarcel Holtmann static bool is_filtered_packet(struct sock *sk, struct sk_buff *skb)
154f81fe64fSMarcel Holtmann {
155f81fe64fSMarcel Holtmann 	struct hci_filter *flt;
156f81fe64fSMarcel Holtmann 	int flt_type, flt_event;
157f81fe64fSMarcel Holtmann 
158f81fe64fSMarcel Holtmann 	/* Apply filter */
159f81fe64fSMarcel Holtmann 	flt = &hci_pi(sk)->filter;
160f81fe64fSMarcel Holtmann 
161d79f34e3SMarcel Holtmann 	flt_type = hci_skb_pkt_type(skb) & HCI_FLT_TYPE_BITS;
162f81fe64fSMarcel Holtmann 
163f81fe64fSMarcel Holtmann 	if (!test_bit(flt_type, &flt->type_mask))
164f81fe64fSMarcel Holtmann 		return true;
165f81fe64fSMarcel Holtmann 
166f81fe64fSMarcel Holtmann 	/* Extra filter for event packets only */
167d79f34e3SMarcel Holtmann 	if (hci_skb_pkt_type(skb) != HCI_EVENT_PKT)
168f81fe64fSMarcel Holtmann 		return false;
169f81fe64fSMarcel Holtmann 
170f81fe64fSMarcel Holtmann 	flt_event = (*(__u8 *)skb->data & HCI_FLT_EVENT_BITS);
171f81fe64fSMarcel Holtmann 
172f81fe64fSMarcel Holtmann 	if (!hci_test_bit(flt_event, &flt->event_mask))
173f81fe64fSMarcel Holtmann 		return true;
174f81fe64fSMarcel Holtmann 
175f81fe64fSMarcel Holtmann 	/* Check filter only when opcode is set */
176f81fe64fSMarcel Holtmann 	if (!flt->opcode)
177f81fe64fSMarcel Holtmann 		return false;
178f81fe64fSMarcel Holtmann 
179f81fe64fSMarcel Holtmann 	if (flt_event == HCI_EV_CMD_COMPLETE &&
180f81fe64fSMarcel Holtmann 	    flt->opcode != get_unaligned((__le16 *)(skb->data + 3)))
181f81fe64fSMarcel Holtmann 		return true;
182f81fe64fSMarcel Holtmann 
183f81fe64fSMarcel Holtmann 	if (flt_event == HCI_EV_CMD_STATUS &&
184f81fe64fSMarcel Holtmann 	    flt->opcode != get_unaligned((__le16 *)(skb->data + 4)))
185f81fe64fSMarcel Holtmann 		return true;
186f81fe64fSMarcel Holtmann 
187f81fe64fSMarcel Holtmann 	return false;
188f81fe64fSMarcel Holtmann }
189f81fe64fSMarcel Holtmann 
1901da177e4SLinus Torvalds /* Send frame to RAW socket */
191470fe1b5SMarcel Holtmann void hci_send_to_sock(struct hci_dev *hdev, struct sk_buff *skb)
1921da177e4SLinus Torvalds {
1931da177e4SLinus Torvalds 	struct sock *sk;
194e0edf373SMarcel Holtmann 	struct sk_buff *skb_copy = NULL;
1951da177e4SLinus Torvalds 
1961da177e4SLinus Torvalds 	BT_DBG("hdev %p len %d", hdev, skb->len);
1971da177e4SLinus Torvalds 
1981da177e4SLinus Torvalds 	read_lock(&hci_sk_list.lock);
199470fe1b5SMarcel Holtmann 
200b67bfe0dSSasha Levin 	sk_for_each(sk, &hci_sk_list.head) {
2011da177e4SLinus Torvalds 		struct sk_buff *nskb;
2021da177e4SLinus Torvalds 
2031da177e4SLinus Torvalds 		if (sk->sk_state != BT_BOUND || hci_pi(sk)->hdev != hdev)
2041da177e4SLinus Torvalds 			continue;
2051da177e4SLinus Torvalds 
2061da177e4SLinus Torvalds 		/* Don't send frame to the socket it came from */
2071da177e4SLinus Torvalds 		if (skb->sk == sk)
2081da177e4SLinus Torvalds 			continue;
2091da177e4SLinus Torvalds 
21023500189SMarcel Holtmann 		if (hci_pi(sk)->channel == HCI_CHANNEL_RAW) {
211d79f34e3SMarcel Holtmann 			if (hci_skb_pkt_type(skb) != HCI_COMMAND_PKT &&
212d79f34e3SMarcel Holtmann 			    hci_skb_pkt_type(skb) != HCI_EVENT_PKT &&
213d79f34e3SMarcel Holtmann 			    hci_skb_pkt_type(skb) != HCI_ACLDATA_PKT &&
214d79f34e3SMarcel Holtmann 			    hci_skb_pkt_type(skb) != HCI_SCODATA_PKT)
215bb77543eSMarcel Holtmann 				continue;
216f81fe64fSMarcel Holtmann 			if (is_filtered_packet(sk, skb))
2171da177e4SLinus Torvalds 				continue;
21823500189SMarcel Holtmann 		} else if (hci_pi(sk)->channel == HCI_CHANNEL_USER) {
21923500189SMarcel Holtmann 			if (!bt_cb(skb)->incoming)
22023500189SMarcel Holtmann 				continue;
221d79f34e3SMarcel Holtmann 			if (hci_skb_pkt_type(skb) != HCI_EVENT_PKT &&
222d79f34e3SMarcel Holtmann 			    hci_skb_pkt_type(skb) != HCI_ACLDATA_PKT &&
223d79f34e3SMarcel Holtmann 			    hci_skb_pkt_type(skb) != HCI_SCODATA_PKT)
22423500189SMarcel Holtmann 				continue;
22523500189SMarcel Holtmann 		} else {
22623500189SMarcel Holtmann 			/* Don't send frame to other channel types */
22723500189SMarcel Holtmann 			continue;
22823500189SMarcel Holtmann 		}
2291da177e4SLinus Torvalds 
230e0edf373SMarcel Holtmann 		if (!skb_copy) {
231e0edf373SMarcel Holtmann 			/* Create a private copy with headroom */
232bad93e9dSOctavian Purdila 			skb_copy = __pskb_copy_fclone(skb, 1, GFP_ATOMIC, true);
233e0edf373SMarcel Holtmann 			if (!skb_copy)
2341da177e4SLinus Torvalds 				continue;
2351da177e4SLinus Torvalds 
2361da177e4SLinus Torvalds 			/* Put type byte before the data */
237d79f34e3SMarcel Holtmann 			memcpy(skb_push(skb_copy, 1), &hci_skb_pkt_type(skb), 1);
238e0edf373SMarcel Holtmann 		}
239e0edf373SMarcel Holtmann 
240e0edf373SMarcel Holtmann 		nskb = skb_clone(skb_copy, GFP_ATOMIC);
241e0edf373SMarcel Holtmann 		if (!nskb)
242e0edf373SMarcel Holtmann 			continue;
2431da177e4SLinus Torvalds 
2441da177e4SLinus Torvalds 		if (sock_queue_rcv_skb(sk, nskb))
2451da177e4SLinus Torvalds 			kfree_skb(nskb);
2461da177e4SLinus Torvalds 	}
247470fe1b5SMarcel Holtmann 
248470fe1b5SMarcel Holtmann 	read_unlock(&hci_sk_list.lock);
249e0edf373SMarcel Holtmann 
250e0edf373SMarcel Holtmann 	kfree_skb(skb_copy);
251470fe1b5SMarcel Holtmann }
252470fe1b5SMarcel Holtmann 
2537129069eSJohan Hedberg /* Send frame to sockets with specific channel */
254a9ee77afSSebastian Andrzej Siewior static void __hci_send_to_channel(unsigned short channel, struct sk_buff *skb,
255c08b1a1dSMarcel Holtmann 				  int flag, struct sock *skip_sk)
256470fe1b5SMarcel Holtmann {
257470fe1b5SMarcel Holtmann 	struct sock *sk;
258470fe1b5SMarcel Holtmann 
2597129069eSJohan Hedberg 	BT_DBG("channel %u len %d", channel, skb->len);
260470fe1b5SMarcel Holtmann 
261b67bfe0dSSasha Levin 	sk_for_each(sk, &hci_sk_list.head) {
262470fe1b5SMarcel Holtmann 		struct sk_buff *nskb;
263470fe1b5SMarcel Holtmann 
264c08b1a1dSMarcel Holtmann 		/* Ignore socket without the flag set */
265c85be545SMarcel Holtmann 		if (!hci_sock_test_flag(sk, flag))
266c08b1a1dSMarcel Holtmann 			continue;
267c08b1a1dSMarcel Holtmann 
268470fe1b5SMarcel Holtmann 		/* Skip the original socket */
269470fe1b5SMarcel Holtmann 		if (sk == skip_sk)
270470fe1b5SMarcel Holtmann 			continue;
271470fe1b5SMarcel Holtmann 
272470fe1b5SMarcel Holtmann 		if (sk->sk_state != BT_BOUND)
273470fe1b5SMarcel Holtmann 			continue;
274470fe1b5SMarcel Holtmann 
2757129069eSJohan Hedberg 		if (hci_pi(sk)->channel != channel)
276d7f72f61SMarcel Holtmann 			continue;
277d7f72f61SMarcel Holtmann 
278d7f72f61SMarcel Holtmann 		nskb = skb_clone(skb, GFP_ATOMIC);
279d7f72f61SMarcel Holtmann 		if (!nskb)
280d7f72f61SMarcel Holtmann 			continue;
281d7f72f61SMarcel Holtmann 
282d7f72f61SMarcel Holtmann 		if (sock_queue_rcv_skb(sk, nskb))
283d7f72f61SMarcel Holtmann 			kfree_skb(nskb);
284d7f72f61SMarcel Holtmann 	}
285d7f72f61SMarcel Holtmann 
286a9ee77afSSebastian Andrzej Siewior }
287a9ee77afSSebastian Andrzej Siewior 
288a9ee77afSSebastian Andrzej Siewior void hci_send_to_channel(unsigned short channel, struct sk_buff *skb,
289a9ee77afSSebastian Andrzej Siewior 			 int flag, struct sock *skip_sk)
290a9ee77afSSebastian Andrzej Siewior {
291a9ee77afSSebastian Andrzej Siewior 	read_lock(&hci_sk_list.lock);
292a9ee77afSSebastian Andrzej Siewior 	__hci_send_to_channel(channel, skb, flag, skip_sk);
293d7f72f61SMarcel Holtmann 	read_unlock(&hci_sk_list.lock);
294d7f72f61SMarcel Holtmann }
295d7f72f61SMarcel Holtmann 
296cd82e61cSMarcel Holtmann /* Send frame to monitor socket */
297cd82e61cSMarcel Holtmann void hci_send_to_monitor(struct hci_dev *hdev, struct sk_buff *skb)
298cd82e61cSMarcel Holtmann {
299cd82e61cSMarcel Holtmann 	struct sk_buff *skb_copy = NULL;
3002b531294SMarcel Holtmann 	struct hci_mon_hdr *hdr;
301cd82e61cSMarcel Holtmann 	__le16 opcode;
302cd82e61cSMarcel Holtmann 
303cd82e61cSMarcel Holtmann 	if (!atomic_read(&monitor_promisc))
304cd82e61cSMarcel Holtmann 		return;
305cd82e61cSMarcel Holtmann 
306cd82e61cSMarcel Holtmann 	BT_DBG("hdev %p len %d", hdev, skb->len);
307cd82e61cSMarcel Holtmann 
308d79f34e3SMarcel Holtmann 	switch (hci_skb_pkt_type(skb)) {
309cd82e61cSMarcel Holtmann 	case HCI_COMMAND_PKT:
310dcf4adbfSJoe Perches 		opcode = cpu_to_le16(HCI_MON_COMMAND_PKT);
311cd82e61cSMarcel Holtmann 		break;
312cd82e61cSMarcel Holtmann 	case HCI_EVENT_PKT:
313dcf4adbfSJoe Perches 		opcode = cpu_to_le16(HCI_MON_EVENT_PKT);
314cd82e61cSMarcel Holtmann 		break;
315cd82e61cSMarcel Holtmann 	case HCI_ACLDATA_PKT:
316cd82e61cSMarcel Holtmann 		if (bt_cb(skb)->incoming)
317dcf4adbfSJoe Perches 			opcode = cpu_to_le16(HCI_MON_ACL_RX_PKT);
318cd82e61cSMarcel Holtmann 		else
319dcf4adbfSJoe Perches 			opcode = cpu_to_le16(HCI_MON_ACL_TX_PKT);
320cd82e61cSMarcel Holtmann 		break;
321cd82e61cSMarcel Holtmann 	case HCI_SCODATA_PKT:
322cd82e61cSMarcel Holtmann 		if (bt_cb(skb)->incoming)
323dcf4adbfSJoe Perches 			opcode = cpu_to_le16(HCI_MON_SCO_RX_PKT);
324cd82e61cSMarcel Holtmann 		else
325dcf4adbfSJoe Perches 			opcode = cpu_to_le16(HCI_MON_SCO_TX_PKT);
326cd82e61cSMarcel Holtmann 		break;
327e875ff84SMarcel Holtmann 	case HCI_DIAG_PKT:
328e875ff84SMarcel Holtmann 		opcode = cpu_to_le16(HCI_MON_VENDOR_DIAG);
329e875ff84SMarcel Holtmann 		break;
330cd82e61cSMarcel Holtmann 	default:
331cd82e61cSMarcel Holtmann 		return;
332cd82e61cSMarcel Holtmann 	}
333cd82e61cSMarcel Holtmann 
3342b531294SMarcel Holtmann 	/* Create a private copy with headroom */
3352b531294SMarcel Holtmann 	skb_copy = __pskb_copy_fclone(skb, HCI_MON_HDR_SIZE, GFP_ATOMIC, true);
3362b531294SMarcel Holtmann 	if (!skb_copy)
3372b531294SMarcel Holtmann 		return;
3382b531294SMarcel Holtmann 
3392b531294SMarcel Holtmann 	/* Put header before the data */
340d58ff351SJohannes Berg 	hdr = skb_push(skb_copy, HCI_MON_HDR_SIZE);
3412b531294SMarcel Holtmann 	hdr->opcode = opcode;
3422b531294SMarcel Holtmann 	hdr->index = cpu_to_le16(hdev->id);
3432b531294SMarcel Holtmann 	hdr->len = cpu_to_le16(skb->len);
3442b531294SMarcel Holtmann 
345c08b1a1dSMarcel Holtmann 	hci_send_to_channel(HCI_CHANNEL_MONITOR, skb_copy,
346c08b1a1dSMarcel Holtmann 			    HCI_SOCK_TRUSTED, NULL);
347cd82e61cSMarcel Holtmann 	kfree_skb(skb_copy);
348cd82e61cSMarcel Holtmann }
349cd82e61cSMarcel Holtmann 
35038ceaa00SMarcel Holtmann void hci_send_monitor_ctrl_event(struct hci_dev *hdev, u16 event,
35138ceaa00SMarcel Holtmann 				 void *data, u16 data_len, ktime_t tstamp,
35238ceaa00SMarcel Holtmann 				 int flag, struct sock *skip_sk)
35338ceaa00SMarcel Holtmann {
35438ceaa00SMarcel Holtmann 	struct sock *sk;
35538ceaa00SMarcel Holtmann 	__le16 index;
35638ceaa00SMarcel Holtmann 
35738ceaa00SMarcel Holtmann 	if (hdev)
35838ceaa00SMarcel Holtmann 		index = cpu_to_le16(hdev->id);
35938ceaa00SMarcel Holtmann 	else
36038ceaa00SMarcel Holtmann 		index = cpu_to_le16(MGMT_INDEX_NONE);
36138ceaa00SMarcel Holtmann 
36238ceaa00SMarcel Holtmann 	read_lock(&hci_sk_list.lock);
36338ceaa00SMarcel Holtmann 
36438ceaa00SMarcel Holtmann 	sk_for_each(sk, &hci_sk_list.head) {
36538ceaa00SMarcel Holtmann 		struct hci_mon_hdr *hdr;
36638ceaa00SMarcel Holtmann 		struct sk_buff *skb;
36738ceaa00SMarcel Holtmann 
36838ceaa00SMarcel Holtmann 		if (hci_pi(sk)->channel != HCI_CHANNEL_CONTROL)
36938ceaa00SMarcel Holtmann 			continue;
37038ceaa00SMarcel Holtmann 
37138ceaa00SMarcel Holtmann 		/* Ignore socket without the flag set */
37238ceaa00SMarcel Holtmann 		if (!hci_sock_test_flag(sk, flag))
37338ceaa00SMarcel Holtmann 			continue;
37438ceaa00SMarcel Holtmann 
37538ceaa00SMarcel Holtmann 		/* Skip the original socket */
37638ceaa00SMarcel Holtmann 		if (sk == skip_sk)
37738ceaa00SMarcel Holtmann 			continue;
37838ceaa00SMarcel Holtmann 
37938ceaa00SMarcel Holtmann 		skb = bt_skb_alloc(6 + data_len, GFP_ATOMIC);
38038ceaa00SMarcel Holtmann 		if (!skb)
38138ceaa00SMarcel Holtmann 			continue;
38238ceaa00SMarcel Holtmann 
38338ceaa00SMarcel Holtmann 		put_unaligned_le32(hci_pi(sk)->cookie, skb_put(skb, 4));
38438ceaa00SMarcel Holtmann 		put_unaligned_le16(event, skb_put(skb, 2));
38538ceaa00SMarcel Holtmann 
38638ceaa00SMarcel Holtmann 		if (data)
38759ae1d12SJohannes Berg 			skb_put_data(skb, data, data_len);
38838ceaa00SMarcel Holtmann 
38938ceaa00SMarcel Holtmann 		skb->tstamp = tstamp;
39038ceaa00SMarcel Holtmann 
391d58ff351SJohannes Berg 		hdr = skb_push(skb, HCI_MON_HDR_SIZE);
39238ceaa00SMarcel Holtmann 		hdr->opcode = cpu_to_le16(HCI_MON_CTRL_EVENT);
39338ceaa00SMarcel Holtmann 		hdr->index = index;
39438ceaa00SMarcel Holtmann 		hdr->len = cpu_to_le16(skb->len - HCI_MON_HDR_SIZE);
39538ceaa00SMarcel Holtmann 
396a9ee77afSSebastian Andrzej Siewior 		__hci_send_to_channel(HCI_CHANNEL_MONITOR, skb,
39738ceaa00SMarcel Holtmann 				      HCI_SOCK_TRUSTED, NULL);
39838ceaa00SMarcel Holtmann 		kfree_skb(skb);
39938ceaa00SMarcel Holtmann 	}
40038ceaa00SMarcel Holtmann 
40138ceaa00SMarcel Holtmann 	read_unlock(&hci_sk_list.lock);
40238ceaa00SMarcel Holtmann }
40338ceaa00SMarcel Holtmann 
404cd82e61cSMarcel Holtmann static struct sk_buff *create_monitor_event(struct hci_dev *hdev, int event)
405cd82e61cSMarcel Holtmann {
406cd82e61cSMarcel Holtmann 	struct hci_mon_hdr *hdr;
407cd82e61cSMarcel Holtmann 	struct hci_mon_new_index *ni;
4086c566dd5SMarcel Holtmann 	struct hci_mon_index_info *ii;
409cd82e61cSMarcel Holtmann 	struct sk_buff *skb;
410cd82e61cSMarcel Holtmann 	__le16 opcode;
411cd82e61cSMarcel Holtmann 
412cd82e61cSMarcel Holtmann 	switch (event) {
413cd82e61cSMarcel Holtmann 	case HCI_DEV_REG:
414cd82e61cSMarcel Holtmann 		skb = bt_skb_alloc(HCI_MON_NEW_INDEX_SIZE, GFP_ATOMIC);
415cd82e61cSMarcel Holtmann 		if (!skb)
416cd82e61cSMarcel Holtmann 			return NULL;
417cd82e61cSMarcel Holtmann 
4184df864c1SJohannes Berg 		ni = skb_put(skb, HCI_MON_NEW_INDEX_SIZE);
419cd82e61cSMarcel Holtmann 		ni->type = hdev->dev_type;
420cd82e61cSMarcel Holtmann 		ni->bus = hdev->bus;
421cd82e61cSMarcel Holtmann 		bacpy(&ni->bdaddr, &hdev->bdaddr);
422cd82e61cSMarcel Holtmann 		memcpy(ni->name, hdev->name, 8);
423cd82e61cSMarcel Holtmann 
424dcf4adbfSJoe Perches 		opcode = cpu_to_le16(HCI_MON_NEW_INDEX);
425cd82e61cSMarcel Holtmann 		break;
426cd82e61cSMarcel Holtmann 
427cd82e61cSMarcel Holtmann 	case HCI_DEV_UNREG:
428cd82e61cSMarcel Holtmann 		skb = bt_skb_alloc(0, GFP_ATOMIC);
429cd82e61cSMarcel Holtmann 		if (!skb)
430cd82e61cSMarcel Holtmann 			return NULL;
431cd82e61cSMarcel Holtmann 
432dcf4adbfSJoe Perches 		opcode = cpu_to_le16(HCI_MON_DEL_INDEX);
433cd82e61cSMarcel Holtmann 		break;
434cd82e61cSMarcel Holtmann 
435e131d74aSMarcel Holtmann 	case HCI_DEV_SETUP:
436e131d74aSMarcel Holtmann 		if (hdev->manufacturer == 0xffff)
437e131d74aSMarcel Holtmann 			return NULL;
438e131d74aSMarcel Holtmann 
439e131d74aSMarcel Holtmann 		/* fall through */
440e131d74aSMarcel Holtmann 
4416c566dd5SMarcel Holtmann 	case HCI_DEV_UP:
4426c566dd5SMarcel Holtmann 		skb = bt_skb_alloc(HCI_MON_INDEX_INFO_SIZE, GFP_ATOMIC);
4436c566dd5SMarcel Holtmann 		if (!skb)
4446c566dd5SMarcel Holtmann 			return NULL;
4456c566dd5SMarcel Holtmann 
4464df864c1SJohannes Berg 		ii = skb_put(skb, HCI_MON_INDEX_INFO_SIZE);
4476c566dd5SMarcel Holtmann 		bacpy(&ii->bdaddr, &hdev->bdaddr);
4486c566dd5SMarcel Holtmann 		ii->manufacturer = cpu_to_le16(hdev->manufacturer);
4496c566dd5SMarcel Holtmann 
4506c566dd5SMarcel Holtmann 		opcode = cpu_to_le16(HCI_MON_INDEX_INFO);
4516c566dd5SMarcel Holtmann 		break;
4526c566dd5SMarcel Holtmann 
45322db3cbcSMarcel Holtmann 	case HCI_DEV_OPEN:
45422db3cbcSMarcel Holtmann 		skb = bt_skb_alloc(0, GFP_ATOMIC);
45522db3cbcSMarcel Holtmann 		if (!skb)
45622db3cbcSMarcel Holtmann 			return NULL;
45722db3cbcSMarcel Holtmann 
45822db3cbcSMarcel Holtmann 		opcode = cpu_to_le16(HCI_MON_OPEN_INDEX);
45922db3cbcSMarcel Holtmann 		break;
46022db3cbcSMarcel Holtmann 
46122db3cbcSMarcel Holtmann 	case HCI_DEV_CLOSE:
46222db3cbcSMarcel Holtmann 		skb = bt_skb_alloc(0, GFP_ATOMIC);
46322db3cbcSMarcel Holtmann 		if (!skb)
46422db3cbcSMarcel Holtmann 			return NULL;
46522db3cbcSMarcel Holtmann 
46622db3cbcSMarcel Holtmann 		opcode = cpu_to_le16(HCI_MON_CLOSE_INDEX);
46722db3cbcSMarcel Holtmann 		break;
46822db3cbcSMarcel Holtmann 
469cd82e61cSMarcel Holtmann 	default:
470cd82e61cSMarcel Holtmann 		return NULL;
471cd82e61cSMarcel Holtmann 	}
472cd82e61cSMarcel Holtmann 
473cd82e61cSMarcel Holtmann 	__net_timestamp(skb);
474cd82e61cSMarcel Holtmann 
475d58ff351SJohannes Berg 	hdr = skb_push(skb, HCI_MON_HDR_SIZE);
476cd82e61cSMarcel Holtmann 	hdr->opcode = opcode;
477cd82e61cSMarcel Holtmann 	hdr->index = cpu_to_le16(hdev->id);
478cd82e61cSMarcel Holtmann 	hdr->len = cpu_to_le16(skb->len - HCI_MON_HDR_SIZE);
479cd82e61cSMarcel Holtmann 
480cd82e61cSMarcel Holtmann 	return skb;
481cd82e61cSMarcel Holtmann }
482cd82e61cSMarcel Holtmann 
483249fa169SMarcel Holtmann static struct sk_buff *create_monitor_ctrl_open(struct sock *sk)
484249fa169SMarcel Holtmann {
485249fa169SMarcel Holtmann 	struct hci_mon_hdr *hdr;
486249fa169SMarcel Holtmann 	struct sk_buff *skb;
487d0bef1d2SMarcel Holtmann 	u16 format;
488249fa169SMarcel Holtmann 	u8 ver[3];
489249fa169SMarcel Holtmann 	u32 flags;
490249fa169SMarcel Holtmann 
4910ef2c42fSMarcel Holtmann 	/* No message needed when cookie is not present */
4920ef2c42fSMarcel Holtmann 	if (!hci_pi(sk)->cookie)
4930ef2c42fSMarcel Holtmann 		return NULL;
4940ef2c42fSMarcel Holtmann 
495d0bef1d2SMarcel Holtmann 	switch (hci_pi(sk)->channel) {
496f81f5b2dSMarcel Holtmann 	case HCI_CHANNEL_RAW:
497f81f5b2dSMarcel Holtmann 		format = 0x0000;
498f81f5b2dSMarcel Holtmann 		ver[0] = BT_SUBSYS_VERSION;
499f81f5b2dSMarcel Holtmann 		put_unaligned_le16(BT_SUBSYS_REVISION, ver + 1);
500f81f5b2dSMarcel Holtmann 		break;
501aa1638ddSMarcel Holtmann 	case HCI_CHANNEL_USER:
502aa1638ddSMarcel Holtmann 		format = 0x0001;
503aa1638ddSMarcel Holtmann 		ver[0] = BT_SUBSYS_VERSION;
504aa1638ddSMarcel Holtmann 		put_unaligned_le16(BT_SUBSYS_REVISION, ver + 1);
505aa1638ddSMarcel Holtmann 		break;
506d0bef1d2SMarcel Holtmann 	case HCI_CHANNEL_CONTROL:
507d0bef1d2SMarcel Holtmann 		format = 0x0002;
508d0bef1d2SMarcel Holtmann 		mgmt_fill_version_info(ver);
509d0bef1d2SMarcel Holtmann 		break;
510d0bef1d2SMarcel Holtmann 	default:
511d0bef1d2SMarcel Holtmann 		/* No message for unsupported format */
512d0bef1d2SMarcel Holtmann 		return NULL;
513d0bef1d2SMarcel Holtmann 	}
514d0bef1d2SMarcel Holtmann 
515249fa169SMarcel Holtmann 	skb = bt_skb_alloc(14 + TASK_COMM_LEN , GFP_ATOMIC);
516249fa169SMarcel Holtmann 	if (!skb)
517249fa169SMarcel Holtmann 		return NULL;
518249fa169SMarcel Holtmann 
519249fa169SMarcel Holtmann 	flags = hci_sock_test_flag(sk, HCI_SOCK_TRUSTED) ? 0x1 : 0x0;
520249fa169SMarcel Holtmann 
521249fa169SMarcel Holtmann 	put_unaligned_le32(hci_pi(sk)->cookie, skb_put(skb, 4));
522249fa169SMarcel Holtmann 	put_unaligned_le16(format, skb_put(skb, 2));
52359ae1d12SJohannes Berg 	skb_put_data(skb, ver, sizeof(ver));
524249fa169SMarcel Holtmann 	put_unaligned_le32(flags, skb_put(skb, 4));
525634fef61SJohannes Berg 	skb_put_u8(skb, TASK_COMM_LEN);
52659ae1d12SJohannes Berg 	skb_put_data(skb, hci_pi(sk)->comm, TASK_COMM_LEN);
527249fa169SMarcel Holtmann 
528249fa169SMarcel Holtmann 	__net_timestamp(skb);
529249fa169SMarcel Holtmann 
530d58ff351SJohannes Berg 	hdr = skb_push(skb, HCI_MON_HDR_SIZE);
531249fa169SMarcel Holtmann 	hdr->opcode = cpu_to_le16(HCI_MON_CTRL_OPEN);
5320ef2c42fSMarcel Holtmann 	if (hci_pi(sk)->hdev)
5330ef2c42fSMarcel Holtmann 		hdr->index = cpu_to_le16(hci_pi(sk)->hdev->id);
5340ef2c42fSMarcel Holtmann 	else
535249fa169SMarcel Holtmann 		hdr->index = cpu_to_le16(HCI_DEV_NONE);
536249fa169SMarcel Holtmann 	hdr->len = cpu_to_le16(skb->len - HCI_MON_HDR_SIZE);
537249fa169SMarcel Holtmann 
538249fa169SMarcel Holtmann 	return skb;
539249fa169SMarcel Holtmann }
540249fa169SMarcel Holtmann 
541249fa169SMarcel Holtmann static struct sk_buff *create_monitor_ctrl_close(struct sock *sk)
542249fa169SMarcel Holtmann {
543249fa169SMarcel Holtmann 	struct hci_mon_hdr *hdr;
544249fa169SMarcel Holtmann 	struct sk_buff *skb;
545249fa169SMarcel Holtmann 
5460ef2c42fSMarcel Holtmann 	/* No message needed when cookie is not present */
5470ef2c42fSMarcel Holtmann 	if (!hci_pi(sk)->cookie)
5480ef2c42fSMarcel Holtmann 		return NULL;
5490ef2c42fSMarcel Holtmann 
550d0bef1d2SMarcel Holtmann 	switch (hci_pi(sk)->channel) {
551f81f5b2dSMarcel Holtmann 	case HCI_CHANNEL_RAW:
552aa1638ddSMarcel Holtmann 	case HCI_CHANNEL_USER:
553d0bef1d2SMarcel Holtmann 	case HCI_CHANNEL_CONTROL:
554d0bef1d2SMarcel Holtmann 		break;
555d0bef1d2SMarcel Holtmann 	default:
556d0bef1d2SMarcel Holtmann 		/* No message for unsupported format */
557d0bef1d2SMarcel Holtmann 		return NULL;
558d0bef1d2SMarcel Holtmann 	}
559d0bef1d2SMarcel Holtmann 
560249fa169SMarcel Holtmann 	skb = bt_skb_alloc(4, GFP_ATOMIC);
561249fa169SMarcel Holtmann 	if (!skb)
562249fa169SMarcel Holtmann 		return NULL;
563249fa169SMarcel Holtmann 
564249fa169SMarcel Holtmann 	put_unaligned_le32(hci_pi(sk)->cookie, skb_put(skb, 4));
565249fa169SMarcel Holtmann 
566249fa169SMarcel Holtmann 	__net_timestamp(skb);
567249fa169SMarcel Holtmann 
568d58ff351SJohannes Berg 	hdr = skb_push(skb, HCI_MON_HDR_SIZE);
569249fa169SMarcel Holtmann 	hdr->opcode = cpu_to_le16(HCI_MON_CTRL_CLOSE);
5700ef2c42fSMarcel Holtmann 	if (hci_pi(sk)->hdev)
5710ef2c42fSMarcel Holtmann 		hdr->index = cpu_to_le16(hci_pi(sk)->hdev->id);
5720ef2c42fSMarcel Holtmann 	else
573249fa169SMarcel Holtmann 		hdr->index = cpu_to_le16(HCI_DEV_NONE);
574249fa169SMarcel Holtmann 	hdr->len = cpu_to_le16(skb->len - HCI_MON_HDR_SIZE);
575249fa169SMarcel Holtmann 
576249fa169SMarcel Holtmann 	return skb;
577249fa169SMarcel Holtmann }
578249fa169SMarcel Holtmann 
57938ceaa00SMarcel Holtmann static struct sk_buff *create_monitor_ctrl_command(struct sock *sk, u16 index,
58038ceaa00SMarcel Holtmann 						   u16 opcode, u16 len,
58138ceaa00SMarcel Holtmann 						   const void *buf)
58238ceaa00SMarcel Holtmann {
58338ceaa00SMarcel Holtmann 	struct hci_mon_hdr *hdr;
58438ceaa00SMarcel Holtmann 	struct sk_buff *skb;
58538ceaa00SMarcel Holtmann 
58638ceaa00SMarcel Holtmann 	skb = bt_skb_alloc(6 + len, GFP_ATOMIC);
58738ceaa00SMarcel Holtmann 	if (!skb)
58838ceaa00SMarcel Holtmann 		return NULL;
58938ceaa00SMarcel Holtmann 
59038ceaa00SMarcel Holtmann 	put_unaligned_le32(hci_pi(sk)->cookie, skb_put(skb, 4));
59138ceaa00SMarcel Holtmann 	put_unaligned_le16(opcode, skb_put(skb, 2));
59238ceaa00SMarcel Holtmann 
59338ceaa00SMarcel Holtmann 	if (buf)
59459ae1d12SJohannes Berg 		skb_put_data(skb, buf, len);
59538ceaa00SMarcel Holtmann 
59638ceaa00SMarcel Holtmann 	__net_timestamp(skb);
59738ceaa00SMarcel Holtmann 
598d58ff351SJohannes Berg 	hdr = skb_push(skb, HCI_MON_HDR_SIZE);
59938ceaa00SMarcel Holtmann 	hdr->opcode = cpu_to_le16(HCI_MON_CTRL_COMMAND);
60038ceaa00SMarcel Holtmann 	hdr->index = cpu_to_le16(index);
60138ceaa00SMarcel Holtmann 	hdr->len = cpu_to_le16(skb->len - HCI_MON_HDR_SIZE);
60238ceaa00SMarcel Holtmann 
60338ceaa00SMarcel Holtmann 	return skb;
60438ceaa00SMarcel Holtmann }
60538ceaa00SMarcel Holtmann 
606787b306cSJohannes Berg static void __printf(2, 3)
607787b306cSJohannes Berg send_monitor_note(struct sock *sk, const char *fmt, ...)
608dd31506dSMarcel Holtmann {
609787b306cSJohannes Berg 	size_t len;
610dd31506dSMarcel Holtmann 	struct hci_mon_hdr *hdr;
611dd31506dSMarcel Holtmann 	struct sk_buff *skb;
612787b306cSJohannes Berg 	va_list args;
613787b306cSJohannes Berg 
614787b306cSJohannes Berg 	va_start(args, fmt);
615787b306cSJohannes Berg 	len = vsnprintf(NULL, 0, fmt, args);
616787b306cSJohannes Berg 	va_end(args);
617dd31506dSMarcel Holtmann 
618dd31506dSMarcel Holtmann 	skb = bt_skb_alloc(len + 1, GFP_ATOMIC);
619dd31506dSMarcel Holtmann 	if (!skb)
620dd31506dSMarcel Holtmann 		return;
621dd31506dSMarcel Holtmann 
622787b306cSJohannes Berg 	va_start(args, fmt);
623787b306cSJohannes Berg 	vsprintf(skb_put(skb, len), fmt, args);
6244df864c1SJohannes Berg 	*(u8 *)skb_put(skb, 1) = 0;
625787b306cSJohannes Berg 	va_end(args);
626dd31506dSMarcel Holtmann 
627dd31506dSMarcel Holtmann 	__net_timestamp(skb);
628dd31506dSMarcel Holtmann 
629dd31506dSMarcel Holtmann 	hdr = (void *)skb_push(skb, HCI_MON_HDR_SIZE);
630dd31506dSMarcel Holtmann 	hdr->opcode = cpu_to_le16(HCI_MON_SYSTEM_NOTE);
631dd31506dSMarcel Holtmann 	hdr->index = cpu_to_le16(HCI_DEV_NONE);
632dd31506dSMarcel Holtmann 	hdr->len = cpu_to_le16(skb->len - HCI_MON_HDR_SIZE);
633dd31506dSMarcel Holtmann 
634dd31506dSMarcel Holtmann 	if (sock_queue_rcv_skb(sk, skb))
635dd31506dSMarcel Holtmann 		kfree_skb(skb);
636dd31506dSMarcel Holtmann }
637dd31506dSMarcel Holtmann 
638cd82e61cSMarcel Holtmann static void send_monitor_replay(struct sock *sk)
639cd82e61cSMarcel Holtmann {
640cd82e61cSMarcel Holtmann 	struct hci_dev *hdev;
641cd82e61cSMarcel Holtmann 
642cd82e61cSMarcel Holtmann 	read_lock(&hci_dev_list_lock);
643cd82e61cSMarcel Holtmann 
644cd82e61cSMarcel Holtmann 	list_for_each_entry(hdev, &hci_dev_list, list) {
645cd82e61cSMarcel Holtmann 		struct sk_buff *skb;
646cd82e61cSMarcel Holtmann 
647cd82e61cSMarcel Holtmann 		skb = create_monitor_event(hdev, HCI_DEV_REG);
648cd82e61cSMarcel Holtmann 		if (!skb)
649cd82e61cSMarcel Holtmann 			continue;
650cd82e61cSMarcel Holtmann 
651cd82e61cSMarcel Holtmann 		if (sock_queue_rcv_skb(sk, skb))
652cd82e61cSMarcel Holtmann 			kfree_skb(skb);
65322db3cbcSMarcel Holtmann 
65422db3cbcSMarcel Holtmann 		if (!test_bit(HCI_RUNNING, &hdev->flags))
65522db3cbcSMarcel Holtmann 			continue;
65622db3cbcSMarcel Holtmann 
65722db3cbcSMarcel Holtmann 		skb = create_monitor_event(hdev, HCI_DEV_OPEN);
65822db3cbcSMarcel Holtmann 		if (!skb)
65922db3cbcSMarcel Holtmann 			continue;
66022db3cbcSMarcel Holtmann 
66122db3cbcSMarcel Holtmann 		if (sock_queue_rcv_skb(sk, skb))
66222db3cbcSMarcel Holtmann 			kfree_skb(skb);
6636c566dd5SMarcel Holtmann 
664e131d74aSMarcel Holtmann 		if (test_bit(HCI_UP, &hdev->flags))
6656c566dd5SMarcel Holtmann 			skb = create_monitor_event(hdev, HCI_DEV_UP);
666e131d74aSMarcel Holtmann 		else if (hci_dev_test_flag(hdev, HCI_SETUP))
667e131d74aSMarcel Holtmann 			skb = create_monitor_event(hdev, HCI_DEV_SETUP);
668e131d74aSMarcel Holtmann 		else
669e131d74aSMarcel Holtmann 			skb = NULL;
6706c566dd5SMarcel Holtmann 
671e131d74aSMarcel Holtmann 		if (skb) {
6726c566dd5SMarcel Holtmann 			if (sock_queue_rcv_skb(sk, skb))
6736c566dd5SMarcel Holtmann 				kfree_skb(skb);
674cd82e61cSMarcel Holtmann 		}
675e131d74aSMarcel Holtmann 	}
676cd82e61cSMarcel Holtmann 
677cd82e61cSMarcel Holtmann 	read_unlock(&hci_dev_list_lock);
678cd82e61cSMarcel Holtmann }
679cd82e61cSMarcel Holtmann 
680249fa169SMarcel Holtmann static void send_monitor_control_replay(struct sock *mon_sk)
681249fa169SMarcel Holtmann {
682249fa169SMarcel Holtmann 	struct sock *sk;
683249fa169SMarcel Holtmann 
684249fa169SMarcel Holtmann 	read_lock(&hci_sk_list.lock);
685249fa169SMarcel Holtmann 
686249fa169SMarcel Holtmann 	sk_for_each(sk, &hci_sk_list.head) {
687249fa169SMarcel Holtmann 		struct sk_buff *skb;
688249fa169SMarcel Holtmann 
689249fa169SMarcel Holtmann 		skb = create_monitor_ctrl_open(sk);
690249fa169SMarcel Holtmann 		if (!skb)
691249fa169SMarcel Holtmann 			continue;
692249fa169SMarcel Holtmann 
693249fa169SMarcel Holtmann 		if (sock_queue_rcv_skb(mon_sk, skb))
694249fa169SMarcel Holtmann 			kfree_skb(skb);
695249fa169SMarcel Holtmann 	}
696249fa169SMarcel Holtmann 
697249fa169SMarcel Holtmann 	read_unlock(&hci_sk_list.lock);
698249fa169SMarcel Holtmann }
699249fa169SMarcel Holtmann 
700040030efSMarcel Holtmann /* Generate internal stack event */
701040030efSMarcel Holtmann static void hci_si_event(struct hci_dev *hdev, int type, int dlen, void *data)
702040030efSMarcel Holtmann {
703040030efSMarcel Holtmann 	struct hci_event_hdr *hdr;
704040030efSMarcel Holtmann 	struct hci_ev_stack_internal *ev;
705040030efSMarcel Holtmann 	struct sk_buff *skb;
706040030efSMarcel Holtmann 
707040030efSMarcel Holtmann 	skb = bt_skb_alloc(HCI_EVENT_HDR_SIZE + sizeof(*ev) + dlen, GFP_ATOMIC);
708040030efSMarcel Holtmann 	if (!skb)
709040030efSMarcel Holtmann 		return;
710040030efSMarcel Holtmann 
7114df864c1SJohannes Berg 	hdr = skb_put(skb, HCI_EVENT_HDR_SIZE);
712040030efSMarcel Holtmann 	hdr->evt  = HCI_EV_STACK_INTERNAL;
713040030efSMarcel Holtmann 	hdr->plen = sizeof(*ev) + dlen;
714040030efSMarcel Holtmann 
7154df864c1SJohannes Berg 	ev = skb_put(skb, sizeof(*ev) + dlen);
716040030efSMarcel Holtmann 	ev->type = type;
717040030efSMarcel Holtmann 	memcpy(ev->data, data, dlen);
718040030efSMarcel Holtmann 
719040030efSMarcel Holtmann 	bt_cb(skb)->incoming = 1;
720040030efSMarcel Holtmann 	__net_timestamp(skb);
721040030efSMarcel Holtmann 
722d79f34e3SMarcel Holtmann 	hci_skb_pkt_type(skb) = HCI_EVENT_PKT;
723040030efSMarcel Holtmann 	hci_send_to_sock(hdev, skb);
724040030efSMarcel Holtmann 	kfree_skb(skb);
725040030efSMarcel Holtmann }
726040030efSMarcel Holtmann 
727040030efSMarcel Holtmann void hci_sock_dev_event(struct hci_dev *hdev, int event)
728040030efSMarcel Holtmann {
729040030efSMarcel Holtmann 	BT_DBG("hdev %s event %d", hdev->name, event);
730040030efSMarcel Holtmann 
731cd82e61cSMarcel Holtmann 	if (atomic_read(&monitor_promisc)) {
732cd82e61cSMarcel Holtmann 		struct sk_buff *skb;
733cd82e61cSMarcel Holtmann 
734ed1b28a4SMarcel Holtmann 		/* Send event to monitor */
735cd82e61cSMarcel Holtmann 		skb = create_monitor_event(hdev, event);
736cd82e61cSMarcel Holtmann 		if (skb) {
737c08b1a1dSMarcel Holtmann 			hci_send_to_channel(HCI_CHANNEL_MONITOR, skb,
738c08b1a1dSMarcel Holtmann 					    HCI_SOCK_TRUSTED, NULL);
739cd82e61cSMarcel Holtmann 			kfree_skb(skb);
740cd82e61cSMarcel Holtmann 		}
741cd82e61cSMarcel Holtmann 	}
742cd82e61cSMarcel Holtmann 
743ed1b28a4SMarcel Holtmann 	if (event <= HCI_DEV_DOWN) {
744ed1b28a4SMarcel Holtmann 		struct hci_ev_si_device ev;
745ed1b28a4SMarcel Holtmann 
746040030efSMarcel Holtmann 		/* Send event to sockets */
747040030efSMarcel Holtmann 		ev.event  = event;
748040030efSMarcel Holtmann 		ev.dev_id = hdev->id;
749040030efSMarcel Holtmann 		hci_si_event(NULL, HCI_EV_SI_DEVICE, sizeof(ev), &ev);
750ed1b28a4SMarcel Holtmann 	}
751040030efSMarcel Holtmann 
752040030efSMarcel Holtmann 	if (event == HCI_DEV_UNREG) {
753040030efSMarcel Holtmann 		struct sock *sk;
754040030efSMarcel Holtmann 
755040030efSMarcel Holtmann 		/* Detach sockets from device */
756040030efSMarcel Holtmann 		read_lock(&hci_sk_list.lock);
757b67bfe0dSSasha Levin 		sk_for_each(sk, &hci_sk_list.head) {
758040030efSMarcel Holtmann 			bh_lock_sock_nested(sk);
759040030efSMarcel Holtmann 			if (hci_pi(sk)->hdev == hdev) {
760040030efSMarcel Holtmann 				hci_pi(sk)->hdev = NULL;
761040030efSMarcel Holtmann 				sk->sk_err = EPIPE;
762040030efSMarcel Holtmann 				sk->sk_state = BT_OPEN;
763040030efSMarcel Holtmann 				sk->sk_state_change(sk);
764040030efSMarcel Holtmann 
765040030efSMarcel Holtmann 				hci_dev_put(hdev);
766040030efSMarcel Holtmann 			}
767040030efSMarcel Holtmann 			bh_unlock_sock(sk);
768040030efSMarcel Holtmann 		}
769040030efSMarcel Holtmann 		read_unlock(&hci_sk_list.lock);
770040030efSMarcel Holtmann 	}
771040030efSMarcel Holtmann }
772040030efSMarcel Holtmann 
773801c1e8dSJohan Hedberg static struct hci_mgmt_chan *__hci_mgmt_chan_find(unsigned short channel)
774801c1e8dSJohan Hedberg {
775801c1e8dSJohan Hedberg 	struct hci_mgmt_chan *c;
776801c1e8dSJohan Hedberg 
777801c1e8dSJohan Hedberg 	list_for_each_entry(c, &mgmt_chan_list, list) {
778801c1e8dSJohan Hedberg 		if (c->channel == channel)
779801c1e8dSJohan Hedberg 			return c;
780801c1e8dSJohan Hedberg 	}
781801c1e8dSJohan Hedberg 
782801c1e8dSJohan Hedberg 	return NULL;
783801c1e8dSJohan Hedberg }
784801c1e8dSJohan Hedberg 
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 	mutex_lock(&mgmt_chan_list_lock);
790801c1e8dSJohan Hedberg 	c = __hci_mgmt_chan_find(channel);
791801c1e8dSJohan Hedberg 	mutex_unlock(&mgmt_chan_list_lock);
792801c1e8dSJohan Hedberg 
793801c1e8dSJohan Hedberg 	return c;
794801c1e8dSJohan Hedberg }
795801c1e8dSJohan Hedberg 
796801c1e8dSJohan Hedberg int hci_mgmt_chan_register(struct hci_mgmt_chan *c)
797801c1e8dSJohan Hedberg {
798801c1e8dSJohan Hedberg 	if (c->channel < HCI_CHANNEL_CONTROL)
799801c1e8dSJohan Hedberg 		return -EINVAL;
800801c1e8dSJohan Hedberg 
801801c1e8dSJohan Hedberg 	mutex_lock(&mgmt_chan_list_lock);
802801c1e8dSJohan Hedberg 	if (__hci_mgmt_chan_find(c->channel)) {
803801c1e8dSJohan Hedberg 		mutex_unlock(&mgmt_chan_list_lock);
804801c1e8dSJohan Hedberg 		return -EALREADY;
805801c1e8dSJohan Hedberg 	}
806801c1e8dSJohan Hedberg 
807801c1e8dSJohan Hedberg 	list_add_tail(&c->list, &mgmt_chan_list);
808801c1e8dSJohan Hedberg 
809801c1e8dSJohan Hedberg 	mutex_unlock(&mgmt_chan_list_lock);
810801c1e8dSJohan Hedberg 
811801c1e8dSJohan Hedberg 	return 0;
812801c1e8dSJohan Hedberg }
813801c1e8dSJohan Hedberg EXPORT_SYMBOL(hci_mgmt_chan_register);
814801c1e8dSJohan Hedberg 
815801c1e8dSJohan Hedberg void hci_mgmt_chan_unregister(struct hci_mgmt_chan *c)
816801c1e8dSJohan Hedberg {
817801c1e8dSJohan Hedberg 	mutex_lock(&mgmt_chan_list_lock);
818801c1e8dSJohan Hedberg 	list_del(&c->list);
819801c1e8dSJohan Hedberg 	mutex_unlock(&mgmt_chan_list_lock);
820801c1e8dSJohan Hedberg }
821801c1e8dSJohan Hedberg EXPORT_SYMBOL(hci_mgmt_chan_unregister);
822801c1e8dSJohan Hedberg 
8231da177e4SLinus Torvalds static int hci_sock_release(struct socket *sock)
8241da177e4SLinus Torvalds {
8251da177e4SLinus Torvalds 	struct sock *sk = sock->sk;
8267b005bd3SMarcel Holtmann 	struct hci_dev *hdev;
827249fa169SMarcel Holtmann 	struct sk_buff *skb;
8281da177e4SLinus Torvalds 
8291da177e4SLinus Torvalds 	BT_DBG("sock %p sk %p", sock, sk);
8301da177e4SLinus Torvalds 
8311da177e4SLinus Torvalds 	if (!sk)
8321da177e4SLinus Torvalds 		return 0;
8331da177e4SLinus Torvalds 
83470ecce91SMarcel Holtmann 	switch (hci_pi(sk)->channel) {
83570ecce91SMarcel Holtmann 	case HCI_CHANNEL_MONITOR:
836cd82e61cSMarcel Holtmann 		atomic_dec(&monitor_promisc);
83770ecce91SMarcel Holtmann 		break;
838f81f5b2dSMarcel Holtmann 	case HCI_CHANNEL_RAW:
839aa1638ddSMarcel Holtmann 	case HCI_CHANNEL_USER:
84070ecce91SMarcel Holtmann 	case HCI_CHANNEL_CONTROL:
841249fa169SMarcel Holtmann 		/* Send event to monitor */
842249fa169SMarcel Holtmann 		skb = create_monitor_ctrl_close(sk);
843249fa169SMarcel Holtmann 		if (skb) {
844249fa169SMarcel Holtmann 			hci_send_to_channel(HCI_CHANNEL_MONITOR, skb,
845249fa169SMarcel Holtmann 					    HCI_SOCK_TRUSTED, NULL);
846249fa169SMarcel Holtmann 			kfree_skb(skb);
847249fa169SMarcel Holtmann 		}
848249fa169SMarcel Holtmann 
849df1cb87aSMarcel Holtmann 		hci_sock_free_cookie(sk);
85070ecce91SMarcel Holtmann 		break;
85170ecce91SMarcel Holtmann 	}
852cd82e61cSMarcel Holtmann 
8531da177e4SLinus Torvalds 	bt_sock_unlink(&hci_sk_list, sk);
8541da177e4SLinus Torvalds 
855e20a2e9cSMyungho Jung 	hdev = hci_pi(sk)->hdev;
8561da177e4SLinus Torvalds 	if (hdev) {
85723500189SMarcel Holtmann 		if (hci_pi(sk)->channel == HCI_CHANNEL_USER) {
8589332ef9dSMasahiro Yamada 			/* When releasing a user channel exclusive access,
8596b3cc1dbSSimon Fels 			 * call hci_dev_do_close directly instead of calling
8606b3cc1dbSSimon Fels 			 * hci_dev_close to ensure the exclusive access will
8616b3cc1dbSSimon Fels 			 * be released and the controller brought back down.
8626b3cc1dbSSimon Fels 			 *
8636b3cc1dbSSimon Fels 			 * The checking of HCI_AUTO_OFF is not needed in this
8646b3cc1dbSSimon Fels 			 * case since it will have been cleared already when
8656b3cc1dbSSimon Fels 			 * opening the user channel.
8666b3cc1dbSSimon Fels 			 */
8676b3cc1dbSSimon Fels 			hci_dev_do_close(hdev);
8689380f9eaSLoic Poulain 			hci_dev_clear_flag(hdev, HCI_USER_CHANNEL);
8699380f9eaSLoic Poulain 			mgmt_index_added(hdev);
87023500189SMarcel Holtmann 		}
87123500189SMarcel Holtmann 
8721da177e4SLinus Torvalds 		atomic_dec(&hdev->promisc);
8731da177e4SLinus Torvalds 		hci_dev_put(hdev);
8741da177e4SLinus Torvalds 	}
8751da177e4SLinus Torvalds 
8761da177e4SLinus Torvalds 	sock_orphan(sk);
8771da177e4SLinus Torvalds 
8781da177e4SLinus Torvalds 	skb_queue_purge(&sk->sk_receive_queue);
8791da177e4SLinus Torvalds 	skb_queue_purge(&sk->sk_write_queue);
8801da177e4SLinus Torvalds 
8811da177e4SLinus Torvalds 	sock_put(sk);
8821da177e4SLinus Torvalds 	return 0;
8831da177e4SLinus Torvalds }
8841da177e4SLinus Torvalds 
885b2a66aadSAntti Julku static int hci_sock_blacklist_add(struct hci_dev *hdev, void __user *arg)
886f0358568SJohan Hedberg {
887f0358568SJohan Hedberg 	bdaddr_t bdaddr;
8885e762444SAntti Julku 	int err;
889f0358568SJohan Hedberg 
890f0358568SJohan Hedberg 	if (copy_from_user(&bdaddr, arg, sizeof(bdaddr)))
891f0358568SJohan Hedberg 		return -EFAULT;
892f0358568SJohan Hedberg 
89309fd0de5SGustavo F. Padovan 	hci_dev_lock(hdev);
8945e762444SAntti Julku 
895dcc36c16SJohan Hedberg 	err = hci_bdaddr_list_add(&hdev->blacklist, &bdaddr, BDADDR_BREDR);
8965e762444SAntti Julku 
89709fd0de5SGustavo F. Padovan 	hci_dev_unlock(hdev);
8985e762444SAntti Julku 
8995e762444SAntti Julku 	return err;
900f0358568SJohan Hedberg }
901f0358568SJohan Hedberg 
902b2a66aadSAntti Julku static int hci_sock_blacklist_del(struct hci_dev *hdev, void __user *arg)
903f0358568SJohan Hedberg {
904f0358568SJohan Hedberg 	bdaddr_t bdaddr;
9055e762444SAntti Julku 	int err;
906f0358568SJohan Hedberg 
907f0358568SJohan Hedberg 	if (copy_from_user(&bdaddr, arg, sizeof(bdaddr)))
908f0358568SJohan Hedberg 		return -EFAULT;
909f0358568SJohan Hedberg 
91009fd0de5SGustavo F. Padovan 	hci_dev_lock(hdev);
9115e762444SAntti Julku 
912dcc36c16SJohan Hedberg 	err = hci_bdaddr_list_del(&hdev->blacklist, &bdaddr, BDADDR_BREDR);
9135e762444SAntti Julku 
91409fd0de5SGustavo F. Padovan 	hci_dev_unlock(hdev);
9155e762444SAntti Julku 
9165e762444SAntti Julku 	return err;
917f0358568SJohan Hedberg }
918f0358568SJohan Hedberg 
9191da177e4SLinus Torvalds /* Ioctls that require bound socket */
9206039aa73SGustavo Padovan static int hci_sock_bound_ioctl(struct sock *sk, unsigned int cmd,
9216039aa73SGustavo Padovan 				unsigned long arg)
9221da177e4SLinus Torvalds {
9231da177e4SLinus Torvalds 	struct hci_dev *hdev = hci_pi(sk)->hdev;
9241da177e4SLinus Torvalds 
9251da177e4SLinus Torvalds 	if (!hdev)
9261da177e4SLinus Torvalds 		return -EBADFD;
9271da177e4SLinus Torvalds 
928d7a5a11dSMarcel Holtmann 	if (hci_dev_test_flag(hdev, HCI_USER_CHANNEL))
9290736cfa8SMarcel Holtmann 		return -EBUSY;
9300736cfa8SMarcel Holtmann 
931d7a5a11dSMarcel Holtmann 	if (hci_dev_test_flag(hdev, HCI_UNCONFIGURED))
932fee746b0SMarcel Holtmann 		return -EOPNOTSUPP;
933fee746b0SMarcel Holtmann 
934ca8bee5dSMarcel Holtmann 	if (hdev->dev_type != HCI_PRIMARY)
9355b69bef5SMarcel Holtmann 		return -EOPNOTSUPP;
9365b69bef5SMarcel Holtmann 
9371da177e4SLinus Torvalds 	switch (cmd) {
9381da177e4SLinus Torvalds 	case HCISETRAW:
9391da177e4SLinus Torvalds 		if (!capable(CAP_NET_ADMIN))
940bf5b30b8SZhao Hongjiang 			return -EPERM;
941db596681SMarcel Holtmann 		return -EOPNOTSUPP;
9421da177e4SLinus Torvalds 
9431da177e4SLinus Torvalds 	case HCIGETCONNINFO:
9441da177e4SLinus Torvalds 		return hci_get_conn_info(hdev, (void __user *)arg);
9451da177e4SLinus Torvalds 
94640be492fSMarcel Holtmann 	case HCIGETAUTHINFO:
94740be492fSMarcel Holtmann 		return hci_get_auth_info(hdev, (void __user *)arg);
94840be492fSMarcel Holtmann 
949f0358568SJohan Hedberg 	case HCIBLOCKADDR:
950f0358568SJohan Hedberg 		if (!capable(CAP_NET_ADMIN))
951bf5b30b8SZhao Hongjiang 			return -EPERM;
952b2a66aadSAntti Julku 		return hci_sock_blacklist_add(hdev, (void __user *)arg);
953f0358568SJohan Hedberg 
954f0358568SJohan Hedberg 	case HCIUNBLOCKADDR:
955f0358568SJohan Hedberg 		if (!capable(CAP_NET_ADMIN))
956bf5b30b8SZhao Hongjiang 			return -EPERM;
957b2a66aadSAntti Julku 		return hci_sock_blacklist_del(hdev, (void __user *)arg);
9580736cfa8SMarcel Holtmann 	}
959f0358568SJohan Hedberg 
960324d36edSMarcel Holtmann 	return -ENOIOCTLCMD;
9611da177e4SLinus Torvalds }
9621da177e4SLinus Torvalds 
9638fc9ced3SGustavo Padovan static int hci_sock_ioctl(struct socket *sock, unsigned int cmd,
9648fc9ced3SGustavo Padovan 			  unsigned long arg)
9651da177e4SLinus Torvalds {
9661da177e4SLinus Torvalds 	void __user *argp = (void __user *)arg;
9670736cfa8SMarcel Holtmann 	struct sock *sk = sock->sk;
9681da177e4SLinus Torvalds 	int err;
9691da177e4SLinus Torvalds 
9701da177e4SLinus Torvalds 	BT_DBG("cmd %x arg %lx", cmd, arg);
9711da177e4SLinus Torvalds 
972c1c4f956SMarcel Holtmann 	lock_sock(sk);
973c1c4f956SMarcel Holtmann 
974c1c4f956SMarcel Holtmann 	if (hci_pi(sk)->channel != HCI_CHANNEL_RAW) {
975c1c4f956SMarcel Holtmann 		err = -EBADFD;
976c1c4f956SMarcel Holtmann 		goto done;
977c1c4f956SMarcel Holtmann 	}
978c1c4f956SMarcel Holtmann 
979f81f5b2dSMarcel Holtmann 	/* When calling an ioctl on an unbound raw socket, then ensure
980f81f5b2dSMarcel Holtmann 	 * that the monitor gets informed. Ensure that the resulting event
981f81f5b2dSMarcel Holtmann 	 * is only send once by checking if the cookie exists or not. The
982f81f5b2dSMarcel Holtmann 	 * socket cookie will be only ever generated once for the lifetime
983f81f5b2dSMarcel Holtmann 	 * of a given socket.
984f81f5b2dSMarcel Holtmann 	 */
985f81f5b2dSMarcel Holtmann 	if (hci_sock_gen_cookie(sk)) {
986f81f5b2dSMarcel Holtmann 		struct sk_buff *skb;
987f81f5b2dSMarcel Holtmann 
988f81f5b2dSMarcel Holtmann 		if (capable(CAP_NET_ADMIN))
989f81f5b2dSMarcel Holtmann 			hci_sock_set_flag(sk, HCI_SOCK_TRUSTED);
990f81f5b2dSMarcel Holtmann 
991f81f5b2dSMarcel Holtmann 		/* Send event to monitor */
992f81f5b2dSMarcel Holtmann 		skb = create_monitor_ctrl_open(sk);
993f81f5b2dSMarcel Holtmann 		if (skb) {
994f81f5b2dSMarcel Holtmann 			hci_send_to_channel(HCI_CHANNEL_MONITOR, skb,
995f81f5b2dSMarcel Holtmann 					    HCI_SOCK_TRUSTED, NULL);
996f81f5b2dSMarcel Holtmann 			kfree_skb(skb);
997f81f5b2dSMarcel Holtmann 		}
998f81f5b2dSMarcel Holtmann 	}
999f81f5b2dSMarcel Holtmann 
1000c1c4f956SMarcel Holtmann 	release_sock(sk);
1001c1c4f956SMarcel Holtmann 
10021da177e4SLinus Torvalds 	switch (cmd) {
10031da177e4SLinus Torvalds 	case HCIGETDEVLIST:
10041da177e4SLinus Torvalds 		return hci_get_dev_list(argp);
10051da177e4SLinus Torvalds 
10061da177e4SLinus Torvalds 	case HCIGETDEVINFO:
10071da177e4SLinus Torvalds 		return hci_get_dev_info(argp);
10081da177e4SLinus Torvalds 
10091da177e4SLinus Torvalds 	case HCIGETCONNLIST:
10101da177e4SLinus Torvalds 		return hci_get_conn_list(argp);
10111da177e4SLinus Torvalds 
10121da177e4SLinus Torvalds 	case HCIDEVUP:
10131da177e4SLinus Torvalds 		if (!capable(CAP_NET_ADMIN))
1014bf5b30b8SZhao Hongjiang 			return -EPERM;
10151da177e4SLinus Torvalds 		return hci_dev_open(arg);
10161da177e4SLinus Torvalds 
10171da177e4SLinus Torvalds 	case HCIDEVDOWN:
10181da177e4SLinus Torvalds 		if (!capable(CAP_NET_ADMIN))
1019bf5b30b8SZhao Hongjiang 			return -EPERM;
10201da177e4SLinus Torvalds 		return hci_dev_close(arg);
10211da177e4SLinus Torvalds 
10221da177e4SLinus Torvalds 	case HCIDEVRESET:
10231da177e4SLinus Torvalds 		if (!capable(CAP_NET_ADMIN))
1024bf5b30b8SZhao Hongjiang 			return -EPERM;
10251da177e4SLinus Torvalds 		return hci_dev_reset(arg);
10261da177e4SLinus Torvalds 
10271da177e4SLinus Torvalds 	case HCIDEVRESTAT:
10281da177e4SLinus Torvalds 		if (!capable(CAP_NET_ADMIN))
1029bf5b30b8SZhao Hongjiang 			return -EPERM;
10301da177e4SLinus Torvalds 		return hci_dev_reset_stat(arg);
10311da177e4SLinus Torvalds 
10321da177e4SLinus Torvalds 	case HCISETSCAN:
10331da177e4SLinus Torvalds 	case HCISETAUTH:
10341da177e4SLinus Torvalds 	case HCISETENCRYPT:
10351da177e4SLinus Torvalds 	case HCISETPTYPE:
10361da177e4SLinus Torvalds 	case HCISETLINKPOL:
10371da177e4SLinus Torvalds 	case HCISETLINKMODE:
10381da177e4SLinus Torvalds 	case HCISETACLMTU:
10391da177e4SLinus Torvalds 	case HCISETSCOMTU:
10401da177e4SLinus Torvalds 		if (!capable(CAP_NET_ADMIN))
1041bf5b30b8SZhao Hongjiang 			return -EPERM;
10421da177e4SLinus Torvalds 		return hci_dev_cmd(cmd, argp);
10431da177e4SLinus Torvalds 
10441da177e4SLinus Torvalds 	case HCIINQUIRY:
10451da177e4SLinus Torvalds 		return hci_inquiry(argp);
1046c1c4f956SMarcel Holtmann 	}
10471da177e4SLinus Torvalds 
10481da177e4SLinus Torvalds 	lock_sock(sk);
1049c1c4f956SMarcel Holtmann 
10501da177e4SLinus Torvalds 	err = hci_sock_bound_ioctl(sk, cmd, arg);
1051c1c4f956SMarcel Holtmann 
1052c1c4f956SMarcel Holtmann done:
10531da177e4SLinus Torvalds 	release_sock(sk);
10541da177e4SLinus Torvalds 	return err;
10551da177e4SLinus Torvalds }
10561da177e4SLinus Torvalds 
1057*7a6038b3SArnd Bergmann #ifdef CONFIG_COMPAT
1058*7a6038b3SArnd Bergmann static int hci_sock_compat_ioctl(struct socket *sock, unsigned int cmd,
1059*7a6038b3SArnd Bergmann 				 unsigned long arg)
1060*7a6038b3SArnd Bergmann {
1061*7a6038b3SArnd Bergmann 	switch (cmd) {
1062*7a6038b3SArnd Bergmann 	case HCIDEVUP:
1063*7a6038b3SArnd Bergmann 	case HCIDEVDOWN:
1064*7a6038b3SArnd Bergmann 	case HCIDEVRESET:
1065*7a6038b3SArnd Bergmann 	case HCIDEVRESTAT:
1066*7a6038b3SArnd Bergmann 		return hci_sock_ioctl(sock, cmd, arg);
1067*7a6038b3SArnd Bergmann 	}
1068*7a6038b3SArnd Bergmann 
1069*7a6038b3SArnd Bergmann 	return hci_sock_ioctl(sock, cmd, (unsigned long)compat_ptr(arg));
1070*7a6038b3SArnd Bergmann }
1071*7a6038b3SArnd Bergmann #endif
1072*7a6038b3SArnd Bergmann 
10738fc9ced3SGustavo Padovan static int hci_sock_bind(struct socket *sock, struct sockaddr *addr,
10748fc9ced3SGustavo Padovan 			 int addr_len)
10751da177e4SLinus Torvalds {
10760381101fSJohan Hedberg 	struct sockaddr_hci haddr;
10771da177e4SLinus Torvalds 	struct sock *sk = sock->sk;
10781da177e4SLinus Torvalds 	struct hci_dev *hdev = NULL;
1079f4cdbb3fSMarcel Holtmann 	struct sk_buff *skb;
10800381101fSJohan Hedberg 	int len, err = 0;
10811da177e4SLinus Torvalds 
10821da177e4SLinus Torvalds 	BT_DBG("sock %p sk %p", sock, sk);
10831da177e4SLinus Torvalds 
10840381101fSJohan Hedberg 	if (!addr)
10850381101fSJohan Hedberg 		return -EINVAL;
10860381101fSJohan Hedberg 
10870381101fSJohan Hedberg 	memset(&haddr, 0, sizeof(haddr));
10880381101fSJohan Hedberg 	len = min_t(unsigned int, sizeof(haddr), addr_len);
10890381101fSJohan Hedberg 	memcpy(&haddr, addr, len);
10900381101fSJohan Hedberg 
10910381101fSJohan Hedberg 	if (haddr.hci_family != AF_BLUETOOTH)
10920381101fSJohan Hedberg 		return -EINVAL;
10930381101fSJohan Hedberg 
10941da177e4SLinus Torvalds 	lock_sock(sk);
10951da177e4SLinus Torvalds 
10967cc2ade2SMarcel Holtmann 	if (sk->sk_state == BT_BOUND) {
10977cc2ade2SMarcel Holtmann 		err = -EALREADY;
10987cc2ade2SMarcel Holtmann 		goto done;
10997cc2ade2SMarcel Holtmann 	}
11007cc2ade2SMarcel Holtmann 
11017cc2ade2SMarcel Holtmann 	switch (haddr.hci_channel) {
11027cc2ade2SMarcel Holtmann 	case HCI_CHANNEL_RAW:
11037cc2ade2SMarcel Holtmann 		if (hci_pi(sk)->hdev) {
11041da177e4SLinus Torvalds 			err = -EALREADY;
11051da177e4SLinus Torvalds 			goto done;
11061da177e4SLinus Torvalds 		}
11071da177e4SLinus Torvalds 
11080381101fSJohan Hedberg 		if (haddr.hci_dev != HCI_DEV_NONE) {
11090381101fSJohan Hedberg 			hdev = hci_dev_get(haddr.hci_dev);
111070f23020SAndrei Emeltchenko 			if (!hdev) {
11111da177e4SLinus Torvalds 				err = -ENODEV;
11121da177e4SLinus Torvalds 				goto done;
11131da177e4SLinus Torvalds 			}
11141da177e4SLinus Torvalds 
11151da177e4SLinus Torvalds 			atomic_inc(&hdev->promisc);
11161da177e4SLinus Torvalds 		}
11171da177e4SLinus Torvalds 
11185a6d2cf5SMarcel Holtmann 		hci_pi(sk)->channel = haddr.hci_channel;
1119f81f5b2dSMarcel Holtmann 
1120f4cdbb3fSMarcel Holtmann 		if (!hci_sock_gen_cookie(sk)) {
1121f4cdbb3fSMarcel Holtmann 			/* In the case when a cookie has already been assigned,
1122f4cdbb3fSMarcel Holtmann 			 * then there has been already an ioctl issued against
1123f4cdbb3fSMarcel Holtmann 			 * an unbound socket and with that triggerd an open
1124f4cdbb3fSMarcel Holtmann 			 * notification. Send a close notification first to
1125f4cdbb3fSMarcel Holtmann 			 * allow the state transition to bounded.
1126f81f5b2dSMarcel Holtmann 			 */
1127f4cdbb3fSMarcel Holtmann 			skb = create_monitor_ctrl_close(sk);
1128f4cdbb3fSMarcel Holtmann 			if (skb) {
1129f4cdbb3fSMarcel Holtmann 				hci_send_to_channel(HCI_CHANNEL_MONITOR, skb,
1130f4cdbb3fSMarcel Holtmann 						    HCI_SOCK_TRUSTED, NULL);
1131f4cdbb3fSMarcel Holtmann 				kfree_skb(skb);
1132f4cdbb3fSMarcel Holtmann 			}
1133f4cdbb3fSMarcel Holtmann 		}
1134f81f5b2dSMarcel Holtmann 
1135f81f5b2dSMarcel Holtmann 		if (capable(CAP_NET_ADMIN))
1136f81f5b2dSMarcel Holtmann 			hci_sock_set_flag(sk, HCI_SOCK_TRUSTED);
1137f81f5b2dSMarcel Holtmann 
1138f4cdbb3fSMarcel Holtmann 		hci_pi(sk)->hdev = hdev;
1139f4cdbb3fSMarcel Holtmann 
1140f81f5b2dSMarcel Holtmann 		/* Send event to monitor */
1141f81f5b2dSMarcel Holtmann 		skb = create_monitor_ctrl_open(sk);
1142f81f5b2dSMarcel Holtmann 		if (skb) {
1143f81f5b2dSMarcel Holtmann 			hci_send_to_channel(HCI_CHANNEL_MONITOR, skb,
1144f81f5b2dSMarcel Holtmann 					    HCI_SOCK_TRUSTED, NULL);
1145f81f5b2dSMarcel Holtmann 			kfree_skb(skb);
1146f81f5b2dSMarcel Holtmann 		}
11477cc2ade2SMarcel Holtmann 		break;
11487cc2ade2SMarcel Holtmann 
114923500189SMarcel Holtmann 	case HCI_CHANNEL_USER:
115023500189SMarcel Holtmann 		if (hci_pi(sk)->hdev) {
115123500189SMarcel Holtmann 			err = -EALREADY;
115223500189SMarcel Holtmann 			goto done;
115323500189SMarcel Holtmann 		}
115423500189SMarcel Holtmann 
115523500189SMarcel Holtmann 		if (haddr.hci_dev == HCI_DEV_NONE) {
115623500189SMarcel Holtmann 			err = -EINVAL;
115723500189SMarcel Holtmann 			goto done;
115823500189SMarcel Holtmann 		}
115923500189SMarcel Holtmann 
116010a8b86fSMarcel Holtmann 		if (!capable(CAP_NET_ADMIN)) {
116123500189SMarcel Holtmann 			err = -EPERM;
116223500189SMarcel Holtmann 			goto done;
116323500189SMarcel Holtmann 		}
116423500189SMarcel Holtmann 
116523500189SMarcel Holtmann 		hdev = hci_dev_get(haddr.hci_dev);
116623500189SMarcel Holtmann 		if (!hdev) {
116723500189SMarcel Holtmann 			err = -ENODEV;
116823500189SMarcel Holtmann 			goto done;
116923500189SMarcel Holtmann 		}
117023500189SMarcel Holtmann 
1171781f899fSMarcel Holtmann 		if (test_bit(HCI_INIT, &hdev->flags) ||
1172d7a5a11dSMarcel Holtmann 		    hci_dev_test_flag(hdev, HCI_SETUP) ||
1173781f899fSMarcel Holtmann 		    hci_dev_test_flag(hdev, HCI_CONFIG) ||
1174781f899fSMarcel Holtmann 		    (!hci_dev_test_flag(hdev, HCI_AUTO_OFF) &&
1175781f899fSMarcel Holtmann 		     test_bit(HCI_UP, &hdev->flags))) {
117623500189SMarcel Holtmann 			err = -EBUSY;
117723500189SMarcel Holtmann 			hci_dev_put(hdev);
117823500189SMarcel Holtmann 			goto done;
117923500189SMarcel Holtmann 		}
118023500189SMarcel Holtmann 
1181238be788SMarcel Holtmann 		if (hci_dev_test_and_set_flag(hdev, HCI_USER_CHANNEL)) {
118223500189SMarcel Holtmann 			err = -EUSERS;
118323500189SMarcel Holtmann 			hci_dev_put(hdev);
118423500189SMarcel Holtmann 			goto done;
118523500189SMarcel Holtmann 		}
118623500189SMarcel Holtmann 
118723500189SMarcel Holtmann 		mgmt_index_removed(hdev);
118823500189SMarcel Holtmann 
118923500189SMarcel Holtmann 		err = hci_dev_open(hdev->id);
119023500189SMarcel Holtmann 		if (err) {
1191781f899fSMarcel Holtmann 			if (err == -EALREADY) {
1192781f899fSMarcel Holtmann 				/* In case the transport is already up and
1193781f899fSMarcel Holtmann 				 * running, clear the error here.
1194781f899fSMarcel Holtmann 				 *
11959332ef9dSMasahiro Yamada 				 * This can happen when opening a user
1196781f899fSMarcel Holtmann 				 * channel and HCI_AUTO_OFF grace period
1197781f899fSMarcel Holtmann 				 * is still active.
1198781f899fSMarcel Holtmann 				 */
1199781f899fSMarcel Holtmann 				err = 0;
1200781f899fSMarcel Holtmann 			} else {
1201a358dc11SMarcel Holtmann 				hci_dev_clear_flag(hdev, HCI_USER_CHANNEL);
1202c6521401SMarcel Holtmann 				mgmt_index_added(hdev);
120323500189SMarcel Holtmann 				hci_dev_put(hdev);
120423500189SMarcel Holtmann 				goto done;
120523500189SMarcel Holtmann 			}
1206781f899fSMarcel Holtmann 		}
120723500189SMarcel Holtmann 
12085a6d2cf5SMarcel Holtmann 		hci_pi(sk)->channel = haddr.hci_channel;
1209aa1638ddSMarcel Holtmann 
1210aa1638ddSMarcel Holtmann 		if (!hci_sock_gen_cookie(sk)) {
1211aa1638ddSMarcel Holtmann 			/* In the case when a cookie has already been assigned,
1212aa1638ddSMarcel Holtmann 			 * this socket will transition from a raw socket into
12139332ef9dSMasahiro Yamada 			 * a user channel socket. For a clean transition, send
1214aa1638ddSMarcel Holtmann 			 * the close notification first.
1215aa1638ddSMarcel Holtmann 			 */
1216aa1638ddSMarcel Holtmann 			skb = create_monitor_ctrl_close(sk);
1217aa1638ddSMarcel Holtmann 			if (skb) {
1218aa1638ddSMarcel Holtmann 				hci_send_to_channel(HCI_CHANNEL_MONITOR, skb,
1219aa1638ddSMarcel Holtmann 						    HCI_SOCK_TRUSTED, NULL);
1220aa1638ddSMarcel Holtmann 				kfree_skb(skb);
1221aa1638ddSMarcel Holtmann 			}
1222aa1638ddSMarcel Holtmann 		}
1223aa1638ddSMarcel Holtmann 
1224aa1638ddSMarcel Holtmann 		/* The user channel is restricted to CAP_NET_ADMIN
1225aa1638ddSMarcel Holtmann 		 * capabilities and with that implicitly trusted.
1226aa1638ddSMarcel Holtmann 		 */
1227aa1638ddSMarcel Holtmann 		hci_sock_set_flag(sk, HCI_SOCK_TRUSTED);
1228aa1638ddSMarcel Holtmann 
122923500189SMarcel Holtmann 		hci_pi(sk)->hdev = hdev;
12305a6d2cf5SMarcel Holtmann 
1231aa1638ddSMarcel Holtmann 		/* Send event to monitor */
1232aa1638ddSMarcel Holtmann 		skb = create_monitor_ctrl_open(sk);
1233aa1638ddSMarcel Holtmann 		if (skb) {
1234aa1638ddSMarcel Holtmann 			hci_send_to_channel(HCI_CHANNEL_MONITOR, skb,
1235aa1638ddSMarcel Holtmann 					    HCI_SOCK_TRUSTED, NULL);
1236aa1638ddSMarcel Holtmann 			kfree_skb(skb);
1237aa1638ddSMarcel Holtmann 		}
1238aa1638ddSMarcel Holtmann 
12395a6d2cf5SMarcel Holtmann 		atomic_inc(&hdev->promisc);
124023500189SMarcel Holtmann 		break;
124123500189SMarcel Holtmann 
1242cd82e61cSMarcel Holtmann 	case HCI_CHANNEL_MONITOR:
1243cd82e61cSMarcel Holtmann 		if (haddr.hci_dev != HCI_DEV_NONE) {
1244cd82e61cSMarcel Holtmann 			err = -EINVAL;
1245cd82e61cSMarcel Holtmann 			goto done;
1246cd82e61cSMarcel Holtmann 		}
1247cd82e61cSMarcel Holtmann 
1248cd82e61cSMarcel Holtmann 		if (!capable(CAP_NET_RAW)) {
1249cd82e61cSMarcel Holtmann 			err = -EPERM;
1250cd82e61cSMarcel Holtmann 			goto done;
1251cd82e61cSMarcel Holtmann 		}
1252cd82e61cSMarcel Holtmann 
12535a6d2cf5SMarcel Holtmann 		hci_pi(sk)->channel = haddr.hci_channel;
12545a6d2cf5SMarcel Holtmann 
125550ebc055SMarcel Holtmann 		/* The monitor interface is restricted to CAP_NET_RAW
125650ebc055SMarcel Holtmann 		 * capabilities and with that implicitly trusted.
125750ebc055SMarcel Holtmann 		 */
125850ebc055SMarcel Holtmann 		hci_sock_set_flag(sk, HCI_SOCK_TRUSTED);
125950ebc055SMarcel Holtmann 
1260787b306cSJohannes Berg 		send_monitor_note(sk, "Linux version %s (%s)",
1261787b306cSJohannes Berg 				  init_utsname()->release,
1262787b306cSJohannes Berg 				  init_utsname()->machine);
12639e8305b3SMarcel Holtmann 		send_monitor_note(sk, "Bluetooth subsystem version %u.%u",
12649e8305b3SMarcel Holtmann 				  BT_SUBSYS_VERSION, BT_SUBSYS_REVISION);
1265cd82e61cSMarcel Holtmann 		send_monitor_replay(sk);
1266249fa169SMarcel Holtmann 		send_monitor_control_replay(sk);
1267cd82e61cSMarcel Holtmann 
1268cd82e61cSMarcel Holtmann 		atomic_inc(&monitor_promisc);
1269cd82e61cSMarcel Holtmann 		break;
1270cd82e61cSMarcel Holtmann 
1271ac714949SMarcel Holtmann 	case HCI_CHANNEL_LOGGING:
1272ac714949SMarcel Holtmann 		if (haddr.hci_dev != HCI_DEV_NONE) {
1273ac714949SMarcel Holtmann 			err = -EINVAL;
1274ac714949SMarcel Holtmann 			goto done;
1275ac714949SMarcel Holtmann 		}
1276ac714949SMarcel Holtmann 
1277ac714949SMarcel Holtmann 		if (!capable(CAP_NET_ADMIN)) {
1278ac714949SMarcel Holtmann 			err = -EPERM;
1279ac714949SMarcel Holtmann 			goto done;
1280ac714949SMarcel Holtmann 		}
12815a6d2cf5SMarcel Holtmann 
12825a6d2cf5SMarcel Holtmann 		hci_pi(sk)->channel = haddr.hci_channel;
1283ac714949SMarcel Holtmann 		break;
1284ac714949SMarcel Holtmann 
12857cc2ade2SMarcel Holtmann 	default:
1286801c1e8dSJohan Hedberg 		if (!hci_mgmt_chan_find(haddr.hci_channel)) {
12877cc2ade2SMarcel Holtmann 			err = -EINVAL;
12887cc2ade2SMarcel Holtmann 			goto done;
12897cc2ade2SMarcel Holtmann 		}
12907cc2ade2SMarcel Holtmann 
1291801c1e8dSJohan Hedberg 		if (haddr.hci_dev != HCI_DEV_NONE) {
1292801c1e8dSJohan Hedberg 			err = -EINVAL;
1293801c1e8dSJohan Hedberg 			goto done;
1294801c1e8dSJohan Hedberg 		}
1295801c1e8dSJohan Hedberg 
12961195fbb8SMarcel Holtmann 		/* Users with CAP_NET_ADMIN capabilities are allowed
12971195fbb8SMarcel Holtmann 		 * access to all management commands and events. For
12981195fbb8SMarcel Holtmann 		 * untrusted users the interface is restricted and
12991195fbb8SMarcel Holtmann 		 * also only untrusted events are sent.
130050ebc055SMarcel Holtmann 		 */
13011195fbb8SMarcel Holtmann 		if (capable(CAP_NET_ADMIN))
130250ebc055SMarcel Holtmann 			hci_sock_set_flag(sk, HCI_SOCK_TRUSTED);
130350ebc055SMarcel Holtmann 
13045a6d2cf5SMarcel Holtmann 		hci_pi(sk)->channel = haddr.hci_channel;
13055a6d2cf5SMarcel Holtmann 
1306f9207338SMarcel Holtmann 		/* At the moment the index and unconfigured index events
1307f9207338SMarcel Holtmann 		 * are enabled unconditionally. Setting them on each
1308f9207338SMarcel Holtmann 		 * socket when binding keeps this functionality. They
1309f9207338SMarcel Holtmann 		 * however might be cleared later and then sending of these
1310f9207338SMarcel Holtmann 		 * events will be disabled, but that is then intentional.
1311f6b7712eSMarcel Holtmann 		 *
1312f6b7712eSMarcel Holtmann 		 * This also enables generic events that are safe to be
1313f6b7712eSMarcel Holtmann 		 * received by untrusted users. Example for such events
1314f6b7712eSMarcel Holtmann 		 * are changes to settings, class of device, name etc.
1315f9207338SMarcel Holtmann 		 */
13165a6d2cf5SMarcel Holtmann 		if (hci_pi(sk)->channel == HCI_CHANNEL_CONTROL) {
1317f4cdbb3fSMarcel Holtmann 			if (!hci_sock_gen_cookie(sk)) {
1318f4cdbb3fSMarcel Holtmann 				/* In the case when a cookie has already been
1319f4cdbb3fSMarcel Holtmann 				 * assigned, this socket will transtion from
1320f4cdbb3fSMarcel Holtmann 				 * a raw socket into a control socket. To
1321f4cdbb3fSMarcel Holtmann 				 * allow for a clean transtion, send the
1322f4cdbb3fSMarcel Holtmann 				 * close notification first.
1323f4cdbb3fSMarcel Holtmann 				 */
1324f4cdbb3fSMarcel Holtmann 				skb = create_monitor_ctrl_close(sk);
1325f4cdbb3fSMarcel Holtmann 				if (skb) {
1326f4cdbb3fSMarcel Holtmann 					hci_send_to_channel(HCI_CHANNEL_MONITOR, skb,
1327f4cdbb3fSMarcel Holtmann 							    HCI_SOCK_TRUSTED, NULL);
1328f4cdbb3fSMarcel Holtmann 					kfree_skb(skb);
1329f4cdbb3fSMarcel Holtmann 				}
1330f4cdbb3fSMarcel Holtmann 			}
133170ecce91SMarcel Holtmann 
1332249fa169SMarcel Holtmann 			/* Send event to monitor */
1333249fa169SMarcel Holtmann 			skb = create_monitor_ctrl_open(sk);
1334249fa169SMarcel Holtmann 			if (skb) {
1335249fa169SMarcel Holtmann 				hci_send_to_channel(HCI_CHANNEL_MONITOR, skb,
1336249fa169SMarcel Holtmann 						    HCI_SOCK_TRUSTED, NULL);
1337249fa169SMarcel Holtmann 				kfree_skb(skb);
1338249fa169SMarcel Holtmann 			}
1339249fa169SMarcel Holtmann 
1340f9207338SMarcel Holtmann 			hci_sock_set_flag(sk, HCI_MGMT_INDEX_EVENTS);
1341f9207338SMarcel Holtmann 			hci_sock_set_flag(sk, HCI_MGMT_UNCONF_INDEX_EVENTS);
13425504c3a3SMarcel Holtmann 			hci_sock_set_flag(sk, HCI_MGMT_OPTION_EVENTS);
13435504c3a3SMarcel Holtmann 			hci_sock_set_flag(sk, HCI_MGMT_SETTING_EVENTS);
13445504c3a3SMarcel Holtmann 			hci_sock_set_flag(sk, HCI_MGMT_DEV_CLASS_EVENTS);
13455504c3a3SMarcel Holtmann 			hci_sock_set_flag(sk, HCI_MGMT_LOCAL_NAME_EVENTS);
1346f9207338SMarcel Holtmann 		}
1347801c1e8dSJohan Hedberg 		break;
1348801c1e8dSJohan Hedberg 	}
1349801c1e8dSJohan Hedberg 
13501da177e4SLinus Torvalds 	sk->sk_state = BT_BOUND;
13511da177e4SLinus Torvalds 
13521da177e4SLinus Torvalds done:
13531da177e4SLinus Torvalds 	release_sock(sk);
13541da177e4SLinus Torvalds 	return err;
13551da177e4SLinus Torvalds }
13561da177e4SLinus Torvalds 
13578fc9ced3SGustavo Padovan static int hci_sock_getname(struct socket *sock, struct sockaddr *addr,
13589b2c45d4SDenys Vlasenko 			    int peer)
13591da177e4SLinus Torvalds {
13601da177e4SLinus Torvalds 	struct sockaddr_hci *haddr = (struct sockaddr_hci *)addr;
13611da177e4SLinus Torvalds 	struct sock *sk = sock->sk;
13629d4b68b2SMarcel Holtmann 	struct hci_dev *hdev;
13639d4b68b2SMarcel Holtmann 	int err = 0;
13641da177e4SLinus Torvalds 
13651da177e4SLinus Torvalds 	BT_DBG("sock %p sk %p", sock, sk);
13661da177e4SLinus Torvalds 
136706f43cbcSMarcel Holtmann 	if (peer)
136806f43cbcSMarcel Holtmann 		return -EOPNOTSUPP;
136906f43cbcSMarcel Holtmann 
13701da177e4SLinus Torvalds 	lock_sock(sk);
13711da177e4SLinus Torvalds 
13729d4b68b2SMarcel Holtmann 	hdev = hci_pi(sk)->hdev;
13739d4b68b2SMarcel Holtmann 	if (!hdev) {
13749d4b68b2SMarcel Holtmann 		err = -EBADFD;
13759d4b68b2SMarcel Holtmann 		goto done;
13769d4b68b2SMarcel Holtmann 	}
13779d4b68b2SMarcel Holtmann 
13781da177e4SLinus Torvalds 	haddr->hci_family = AF_BLUETOOTH;
13797b005bd3SMarcel Holtmann 	haddr->hci_dev    = hdev->id;
13809d4b68b2SMarcel Holtmann 	haddr->hci_channel= hci_pi(sk)->channel;
13819b2c45d4SDenys Vlasenko 	err = sizeof(*haddr);
13821da177e4SLinus Torvalds 
13839d4b68b2SMarcel Holtmann done:
13841da177e4SLinus Torvalds 	release_sock(sk);
13859d4b68b2SMarcel Holtmann 	return err;
13861da177e4SLinus Torvalds }
13871da177e4SLinus Torvalds 
13886039aa73SGustavo Padovan static void hci_sock_cmsg(struct sock *sk, struct msghdr *msg,
13896039aa73SGustavo Padovan 			  struct sk_buff *skb)
13901da177e4SLinus Torvalds {
13911da177e4SLinus Torvalds 	__u32 mask = hci_pi(sk)->cmsg_mask;
13921da177e4SLinus Torvalds 
13930d48d939SMarcel Holtmann 	if (mask & HCI_CMSG_DIR) {
13940d48d939SMarcel Holtmann 		int incoming = bt_cb(skb)->incoming;
13958fc9ced3SGustavo Padovan 		put_cmsg(msg, SOL_HCI, HCI_CMSG_DIR, sizeof(incoming),
13968fc9ced3SGustavo Padovan 			 &incoming);
13970d48d939SMarcel Holtmann 	}
13981da177e4SLinus Torvalds 
1399a61bbcf2SPatrick McHardy 	if (mask & HCI_CMSG_TSTAMP) {
1400f6e623a6SJohann Felix Soden #ifdef CONFIG_COMPAT
140113c6ee2aSDeepa Dinamani 		struct old_timeval32 ctv;
1402f6e623a6SJohann Felix Soden #endif
140313c6ee2aSDeepa Dinamani 		struct __kernel_old_timeval tv;
1404767c5eb5SMarcel Holtmann 		void *data;
1405767c5eb5SMarcel Holtmann 		int len;
1406a61bbcf2SPatrick McHardy 
1407a61bbcf2SPatrick McHardy 		skb_get_timestamp(skb, &tv);
1408767c5eb5SMarcel Holtmann 
14091da97f83SDavid S. Miller 		data = &tv;
14101da97f83SDavid S. Miller 		len = sizeof(tv);
14111da97f83SDavid S. Miller #ifdef CONFIG_COMPAT
1412da88cea1SH. J. Lu 		if (!COMPAT_USE_64BIT_TIME &&
1413da88cea1SH. J. Lu 		    (msg->msg_flags & MSG_CMSG_COMPAT)) {
1414767c5eb5SMarcel Holtmann 			ctv.tv_sec = tv.tv_sec;
1415767c5eb5SMarcel Holtmann 			ctv.tv_usec = tv.tv_usec;
1416767c5eb5SMarcel Holtmann 			data = &ctv;
1417767c5eb5SMarcel Holtmann 			len = sizeof(ctv);
1418767c5eb5SMarcel Holtmann 		}
14191da97f83SDavid S. Miller #endif
1420767c5eb5SMarcel Holtmann 
1421767c5eb5SMarcel Holtmann 		put_cmsg(msg, SOL_HCI, HCI_CMSG_TSTAMP, len, data);
1422a61bbcf2SPatrick McHardy 	}
14231da177e4SLinus Torvalds }
14241da177e4SLinus Torvalds 
14258528d3f7SMarcel Holtmann static int hci_sock_recvmsg(struct socket *sock, struct msghdr *msg,
14268528d3f7SMarcel Holtmann 			    size_t len, int flags)
14271da177e4SLinus Torvalds {
14281da177e4SLinus Torvalds 	int noblock = flags & MSG_DONTWAIT;
14291da177e4SLinus Torvalds 	struct sock *sk = sock->sk;
14301da177e4SLinus Torvalds 	struct sk_buff *skb;
14311da177e4SLinus Torvalds 	int copied, err;
143283871f8cSDenis Kenzior 	unsigned int skblen;
14331da177e4SLinus Torvalds 
14341da177e4SLinus Torvalds 	BT_DBG("sock %p, sk %p", sock, sk);
14351da177e4SLinus Torvalds 
1436d94a6104SMarcel Holtmann 	if (flags & MSG_OOB)
14371da177e4SLinus Torvalds 		return -EOPNOTSUPP;
14381da177e4SLinus Torvalds 
1439ac714949SMarcel Holtmann 	if (hci_pi(sk)->channel == HCI_CHANNEL_LOGGING)
1440ac714949SMarcel Holtmann 		return -EOPNOTSUPP;
1441ac714949SMarcel Holtmann 
14421da177e4SLinus Torvalds 	if (sk->sk_state == BT_CLOSED)
14431da177e4SLinus Torvalds 		return 0;
14441da177e4SLinus Torvalds 
144570f23020SAndrei Emeltchenko 	skb = skb_recv_datagram(sk, flags, noblock, &err);
144670f23020SAndrei Emeltchenko 	if (!skb)
14471da177e4SLinus Torvalds 		return err;
14481da177e4SLinus Torvalds 
144983871f8cSDenis Kenzior 	skblen = skb->len;
14501da177e4SLinus Torvalds 	copied = skb->len;
14511da177e4SLinus Torvalds 	if (len < copied) {
14521da177e4SLinus Torvalds 		msg->msg_flags |= MSG_TRUNC;
14531da177e4SLinus Torvalds 		copied = len;
14541da177e4SLinus Torvalds 	}
14551da177e4SLinus Torvalds 
1456badff6d0SArnaldo Carvalho de Melo 	skb_reset_transport_header(skb);
145751f3d02bSDavid S. Miller 	err = skb_copy_datagram_msg(skb, 0, msg, copied);
14581da177e4SLinus Torvalds 
14593a208627SMarcel Holtmann 	switch (hci_pi(sk)->channel) {
14603a208627SMarcel Holtmann 	case HCI_CHANNEL_RAW:
14611da177e4SLinus Torvalds 		hci_sock_cmsg(sk, msg, skb);
14623a208627SMarcel Holtmann 		break;
146323500189SMarcel Holtmann 	case HCI_CHANNEL_USER:
1464cd82e61cSMarcel Holtmann 	case HCI_CHANNEL_MONITOR:
1465cd82e61cSMarcel Holtmann 		sock_recv_timestamp(msg, sk, skb);
1466cd82e61cSMarcel Holtmann 		break;
1467801c1e8dSJohan Hedberg 	default:
1468801c1e8dSJohan Hedberg 		if (hci_mgmt_chan_find(hci_pi(sk)->channel))
1469801c1e8dSJohan Hedberg 			sock_recv_timestamp(msg, sk, skb);
1470801c1e8dSJohan Hedberg 		break;
14713a208627SMarcel Holtmann 	}
14721da177e4SLinus Torvalds 
14731da177e4SLinus Torvalds 	skb_free_datagram(sk, skb);
14741da177e4SLinus Torvalds 
14754f34228bSLuiz Augusto von Dentz 	if (flags & MSG_TRUNC)
147683871f8cSDenis Kenzior 		copied = skblen;
147783871f8cSDenis Kenzior 
14781da177e4SLinus Torvalds 	return err ? : copied;
14791da177e4SLinus Torvalds }
14801da177e4SLinus Torvalds 
1481fa4335d7SJohan Hedberg static int hci_mgmt_cmd(struct hci_mgmt_chan *chan, struct sock *sk,
1482fa4335d7SJohan Hedberg 			struct msghdr *msg, size_t msglen)
1483fa4335d7SJohan Hedberg {
1484fa4335d7SJohan Hedberg 	void *buf;
1485fa4335d7SJohan Hedberg 	u8 *cp;
1486fa4335d7SJohan Hedberg 	struct mgmt_hdr *hdr;
1487fa4335d7SJohan Hedberg 	u16 opcode, index, len;
1488fa4335d7SJohan Hedberg 	struct hci_dev *hdev = NULL;
1489fa4335d7SJohan Hedberg 	const struct hci_mgmt_handler *handler;
1490fa4335d7SJohan Hedberg 	bool var_len, no_hdev;
1491fa4335d7SJohan Hedberg 	int err;
1492fa4335d7SJohan Hedberg 
1493fa4335d7SJohan Hedberg 	BT_DBG("got %zu bytes", msglen);
1494fa4335d7SJohan Hedberg 
1495fa4335d7SJohan Hedberg 	if (msglen < sizeof(*hdr))
1496fa4335d7SJohan Hedberg 		return -EINVAL;
1497fa4335d7SJohan Hedberg 
1498fa4335d7SJohan Hedberg 	buf = kmalloc(msglen, GFP_KERNEL);
1499fa4335d7SJohan Hedberg 	if (!buf)
1500fa4335d7SJohan Hedberg 		return -ENOMEM;
1501fa4335d7SJohan Hedberg 
1502fa4335d7SJohan Hedberg 	if (memcpy_from_msg(buf, msg, msglen)) {
1503fa4335d7SJohan Hedberg 		err = -EFAULT;
1504fa4335d7SJohan Hedberg 		goto done;
1505fa4335d7SJohan Hedberg 	}
1506fa4335d7SJohan Hedberg 
1507fa4335d7SJohan Hedberg 	hdr = buf;
1508fa4335d7SJohan Hedberg 	opcode = __le16_to_cpu(hdr->opcode);
1509fa4335d7SJohan Hedberg 	index = __le16_to_cpu(hdr->index);
1510fa4335d7SJohan Hedberg 	len = __le16_to_cpu(hdr->len);
1511fa4335d7SJohan Hedberg 
1512fa4335d7SJohan Hedberg 	if (len != msglen - sizeof(*hdr)) {
1513fa4335d7SJohan Hedberg 		err = -EINVAL;
1514fa4335d7SJohan Hedberg 		goto done;
1515fa4335d7SJohan Hedberg 	}
1516fa4335d7SJohan Hedberg 
151738ceaa00SMarcel Holtmann 	if (chan->channel == HCI_CHANNEL_CONTROL) {
151838ceaa00SMarcel Holtmann 		struct sk_buff *skb;
151938ceaa00SMarcel Holtmann 
152038ceaa00SMarcel Holtmann 		/* Send event to monitor */
152138ceaa00SMarcel Holtmann 		skb = create_monitor_ctrl_command(sk, index, opcode, len,
152238ceaa00SMarcel Holtmann 						  buf + sizeof(*hdr));
152338ceaa00SMarcel Holtmann 		if (skb) {
152438ceaa00SMarcel Holtmann 			hci_send_to_channel(HCI_CHANNEL_MONITOR, skb,
152538ceaa00SMarcel Holtmann 					    HCI_SOCK_TRUSTED, NULL);
152638ceaa00SMarcel Holtmann 			kfree_skb(skb);
152738ceaa00SMarcel Holtmann 		}
152838ceaa00SMarcel Holtmann 	}
152938ceaa00SMarcel Holtmann 
1530fa4335d7SJohan Hedberg 	if (opcode >= chan->handler_count ||
1531fa4335d7SJohan Hedberg 	    chan->handlers[opcode].func == NULL) {
1532fa4335d7SJohan Hedberg 		BT_DBG("Unknown op %u", opcode);
1533fa4335d7SJohan Hedberg 		err = mgmt_cmd_status(sk, index, opcode,
1534fa4335d7SJohan Hedberg 				      MGMT_STATUS_UNKNOWN_COMMAND);
1535fa4335d7SJohan Hedberg 		goto done;
1536fa4335d7SJohan Hedberg 	}
1537fa4335d7SJohan Hedberg 
1538fa4335d7SJohan Hedberg 	handler = &chan->handlers[opcode];
1539fa4335d7SJohan Hedberg 
1540fa4335d7SJohan Hedberg 	if (!hci_sock_test_flag(sk, HCI_SOCK_TRUSTED) &&
1541fa4335d7SJohan Hedberg 	    !(handler->flags & HCI_MGMT_UNTRUSTED)) {
1542fa4335d7SJohan Hedberg 		err = mgmt_cmd_status(sk, index, opcode,
1543fa4335d7SJohan Hedberg 				      MGMT_STATUS_PERMISSION_DENIED);
1544fa4335d7SJohan Hedberg 		goto done;
1545fa4335d7SJohan Hedberg 	}
1546fa4335d7SJohan Hedberg 
1547fa4335d7SJohan Hedberg 	if (index != MGMT_INDEX_NONE) {
1548fa4335d7SJohan Hedberg 		hdev = hci_dev_get(index);
1549fa4335d7SJohan Hedberg 		if (!hdev) {
1550fa4335d7SJohan Hedberg 			err = mgmt_cmd_status(sk, index, opcode,
1551fa4335d7SJohan Hedberg 					      MGMT_STATUS_INVALID_INDEX);
1552fa4335d7SJohan Hedberg 			goto done;
1553fa4335d7SJohan Hedberg 		}
1554fa4335d7SJohan Hedberg 
1555fa4335d7SJohan Hedberg 		if (hci_dev_test_flag(hdev, HCI_SETUP) ||
1556fa4335d7SJohan Hedberg 		    hci_dev_test_flag(hdev, HCI_CONFIG) ||
1557fa4335d7SJohan Hedberg 		    hci_dev_test_flag(hdev, HCI_USER_CHANNEL)) {
1558fa4335d7SJohan Hedberg 			err = mgmt_cmd_status(sk, index, opcode,
1559fa4335d7SJohan Hedberg 					      MGMT_STATUS_INVALID_INDEX);
1560fa4335d7SJohan Hedberg 			goto done;
1561fa4335d7SJohan Hedberg 		}
1562fa4335d7SJohan Hedberg 
1563fa4335d7SJohan Hedberg 		if (hci_dev_test_flag(hdev, HCI_UNCONFIGURED) &&
1564fa4335d7SJohan Hedberg 		    !(handler->flags & HCI_MGMT_UNCONFIGURED)) {
1565fa4335d7SJohan Hedberg 			err = mgmt_cmd_status(sk, index, opcode,
1566fa4335d7SJohan Hedberg 					      MGMT_STATUS_INVALID_INDEX);
1567fa4335d7SJohan Hedberg 			goto done;
1568fa4335d7SJohan Hedberg 		}
1569fa4335d7SJohan Hedberg 	}
1570fa4335d7SJohan Hedberg 
1571fa4335d7SJohan Hedberg 	no_hdev = (handler->flags & HCI_MGMT_NO_HDEV);
1572fa4335d7SJohan Hedberg 	if (no_hdev != !hdev) {
1573fa4335d7SJohan Hedberg 		err = mgmt_cmd_status(sk, index, opcode,
1574fa4335d7SJohan Hedberg 				      MGMT_STATUS_INVALID_INDEX);
1575fa4335d7SJohan Hedberg 		goto done;
1576fa4335d7SJohan Hedberg 	}
1577fa4335d7SJohan Hedberg 
1578fa4335d7SJohan Hedberg 	var_len = (handler->flags & HCI_MGMT_VAR_LEN);
1579fa4335d7SJohan Hedberg 	if ((var_len && len < handler->data_len) ||
1580fa4335d7SJohan Hedberg 	    (!var_len && len != handler->data_len)) {
1581fa4335d7SJohan Hedberg 		err = mgmt_cmd_status(sk, index, opcode,
1582fa4335d7SJohan Hedberg 				      MGMT_STATUS_INVALID_PARAMS);
1583fa4335d7SJohan Hedberg 		goto done;
1584fa4335d7SJohan Hedberg 	}
1585fa4335d7SJohan Hedberg 
1586fa4335d7SJohan Hedberg 	if (hdev && chan->hdev_init)
1587fa4335d7SJohan Hedberg 		chan->hdev_init(sk, hdev);
1588fa4335d7SJohan Hedberg 
1589fa4335d7SJohan Hedberg 	cp = buf + sizeof(*hdr);
1590fa4335d7SJohan Hedberg 
1591fa4335d7SJohan Hedberg 	err = handler->func(sk, hdev, cp, len);
1592fa4335d7SJohan Hedberg 	if (err < 0)
1593fa4335d7SJohan Hedberg 		goto done;
1594fa4335d7SJohan Hedberg 
1595fa4335d7SJohan Hedberg 	err = msglen;
1596fa4335d7SJohan Hedberg 
1597fa4335d7SJohan Hedberg done:
1598fa4335d7SJohan Hedberg 	if (hdev)
1599fa4335d7SJohan Hedberg 		hci_dev_put(hdev);
1600fa4335d7SJohan Hedberg 
1601fa4335d7SJohan Hedberg 	kfree(buf);
1602fa4335d7SJohan Hedberg 	return err;
1603fa4335d7SJohan Hedberg }
1604fa4335d7SJohan Hedberg 
1605ac714949SMarcel Holtmann static int hci_logging_frame(struct sock *sk, struct msghdr *msg, int len)
1606ac714949SMarcel Holtmann {
1607ac714949SMarcel Holtmann 	struct hci_mon_hdr *hdr;
1608ac714949SMarcel Holtmann 	struct sk_buff *skb;
1609ac714949SMarcel Holtmann 	struct hci_dev *hdev;
1610ac714949SMarcel Holtmann 	u16 index;
1611ac714949SMarcel Holtmann 	int err;
1612ac714949SMarcel Holtmann 
1613ac714949SMarcel Holtmann 	/* The logging frame consists at minimum of the standard header,
1614ac714949SMarcel Holtmann 	 * the priority byte, the ident length byte and at least one string
1615ac714949SMarcel Holtmann 	 * terminator NUL byte. Anything shorter are invalid packets.
1616ac714949SMarcel Holtmann 	 */
1617ac714949SMarcel Holtmann 	if (len < sizeof(*hdr) + 3)
1618ac714949SMarcel Holtmann 		return -EINVAL;
1619ac714949SMarcel Holtmann 
1620ac714949SMarcel Holtmann 	skb = bt_skb_send_alloc(sk, len, msg->msg_flags & MSG_DONTWAIT, &err);
1621ac714949SMarcel Holtmann 	if (!skb)
1622ac714949SMarcel Holtmann 		return err;
1623ac714949SMarcel Holtmann 
1624ac714949SMarcel Holtmann 	if (memcpy_from_msg(skb_put(skb, len), msg, len)) {
1625ac714949SMarcel Holtmann 		err = -EFAULT;
1626ac714949SMarcel Holtmann 		goto drop;
1627ac714949SMarcel Holtmann 	}
1628ac714949SMarcel Holtmann 
1629ac714949SMarcel Holtmann 	hdr = (void *)skb->data;
1630ac714949SMarcel Holtmann 
1631ac714949SMarcel Holtmann 	if (__le16_to_cpu(hdr->len) != len - sizeof(*hdr)) {
1632ac714949SMarcel Holtmann 		err = -EINVAL;
1633ac714949SMarcel Holtmann 		goto drop;
1634ac714949SMarcel Holtmann 	}
1635ac714949SMarcel Holtmann 
1636ac714949SMarcel Holtmann 	if (__le16_to_cpu(hdr->opcode) == 0x0000) {
1637ac714949SMarcel Holtmann 		__u8 priority = skb->data[sizeof(*hdr)];
1638ac714949SMarcel Holtmann 		__u8 ident_len = skb->data[sizeof(*hdr) + 1];
1639ac714949SMarcel Holtmann 
1640ac714949SMarcel Holtmann 		/* Only the priorities 0-7 are valid and with that any other
1641ac714949SMarcel Holtmann 		 * value results in an invalid packet.
1642ac714949SMarcel Holtmann 		 *
1643ac714949SMarcel Holtmann 		 * The priority byte is followed by an ident length byte and
1644ac714949SMarcel Holtmann 		 * the NUL terminated ident string. Check that the ident
1645ac714949SMarcel Holtmann 		 * length is not overflowing the packet and also that the
1646ac714949SMarcel Holtmann 		 * ident string itself is NUL terminated. In case the ident
1647ac714949SMarcel Holtmann 		 * length is zero, the length value actually doubles as NUL
1648ac714949SMarcel Holtmann 		 * terminator identifier.
1649ac714949SMarcel Holtmann 		 *
1650ac714949SMarcel Holtmann 		 * The message follows the ident string (if present) and
1651ac714949SMarcel Holtmann 		 * must be NUL terminated. Otherwise it is not a valid packet.
1652ac714949SMarcel Holtmann 		 */
1653ac714949SMarcel Holtmann 		if (priority > 7 || skb->data[len - 1] != 0x00 ||
1654ac714949SMarcel Holtmann 		    ident_len > len - sizeof(*hdr) - 3 ||
1655ac714949SMarcel Holtmann 		    skb->data[sizeof(*hdr) + ident_len + 1] != 0x00) {
1656ac714949SMarcel Holtmann 			err = -EINVAL;
1657ac714949SMarcel Holtmann 			goto drop;
1658ac714949SMarcel Holtmann 		}
1659ac714949SMarcel Holtmann 	} else {
1660ac714949SMarcel Holtmann 		err = -EINVAL;
1661ac714949SMarcel Holtmann 		goto drop;
1662ac714949SMarcel Holtmann 	}
1663ac714949SMarcel Holtmann 
1664ac714949SMarcel Holtmann 	index = __le16_to_cpu(hdr->index);
1665ac714949SMarcel Holtmann 
1666ac714949SMarcel Holtmann 	if (index != MGMT_INDEX_NONE) {
1667ac714949SMarcel Holtmann 		hdev = hci_dev_get(index);
1668ac714949SMarcel Holtmann 		if (!hdev) {
1669ac714949SMarcel Holtmann 			err = -ENODEV;
1670ac714949SMarcel Holtmann 			goto drop;
1671ac714949SMarcel Holtmann 		}
1672ac714949SMarcel Holtmann 	} else {
1673ac714949SMarcel Holtmann 		hdev = NULL;
1674ac714949SMarcel Holtmann 	}
1675ac714949SMarcel Holtmann 
1676ac714949SMarcel Holtmann 	hdr->opcode = cpu_to_le16(HCI_MON_USER_LOGGING);
1677ac714949SMarcel Holtmann 
1678ac714949SMarcel Holtmann 	hci_send_to_channel(HCI_CHANNEL_MONITOR, skb, HCI_SOCK_TRUSTED, NULL);
1679ac714949SMarcel Holtmann 	err = len;
1680ac714949SMarcel Holtmann 
1681ac714949SMarcel Holtmann 	if (hdev)
1682ac714949SMarcel Holtmann 		hci_dev_put(hdev);
1683ac714949SMarcel Holtmann 
1684ac714949SMarcel Holtmann drop:
1685ac714949SMarcel Holtmann 	kfree_skb(skb);
1686ac714949SMarcel Holtmann 	return err;
1687ac714949SMarcel Holtmann }
1688ac714949SMarcel Holtmann 
16891b784140SYing Xue static int hci_sock_sendmsg(struct socket *sock, struct msghdr *msg,
16901b784140SYing Xue 			    size_t len)
16911da177e4SLinus Torvalds {
16921da177e4SLinus Torvalds 	struct sock *sk = sock->sk;
1693801c1e8dSJohan Hedberg 	struct hci_mgmt_chan *chan;
16941da177e4SLinus Torvalds 	struct hci_dev *hdev;
16951da177e4SLinus Torvalds 	struct sk_buff *skb;
16961da177e4SLinus Torvalds 	int err;
16971da177e4SLinus Torvalds 
16981da177e4SLinus Torvalds 	BT_DBG("sock %p sk %p", sock, sk);
16991da177e4SLinus Torvalds 
17001da177e4SLinus Torvalds 	if (msg->msg_flags & MSG_OOB)
17011da177e4SLinus Torvalds 		return -EOPNOTSUPP;
17021da177e4SLinus Torvalds 
1703ab89f0bdSSzymon Janc 	if (msg->msg_flags & ~(MSG_DONTWAIT|MSG_NOSIGNAL|MSG_ERRQUEUE|
1704ab89f0bdSSzymon Janc 			       MSG_CMSG_COMPAT))
17051da177e4SLinus Torvalds 		return -EINVAL;
17061da177e4SLinus Torvalds 
17071da177e4SLinus Torvalds 	if (len < 4 || len > HCI_MAX_FRAME_SIZE)
17081da177e4SLinus Torvalds 		return -EINVAL;
17091da177e4SLinus Torvalds 
17101da177e4SLinus Torvalds 	lock_sock(sk);
17111da177e4SLinus Torvalds 
17120381101fSJohan Hedberg 	switch (hci_pi(sk)->channel) {
17130381101fSJohan Hedberg 	case HCI_CHANNEL_RAW:
171423500189SMarcel Holtmann 	case HCI_CHANNEL_USER:
17150381101fSJohan Hedberg 		break;
1716cd82e61cSMarcel Holtmann 	case HCI_CHANNEL_MONITOR:
1717cd82e61cSMarcel Holtmann 		err = -EOPNOTSUPP;
1718cd82e61cSMarcel Holtmann 		goto done;
1719ac714949SMarcel Holtmann 	case HCI_CHANNEL_LOGGING:
1720ac714949SMarcel Holtmann 		err = hci_logging_frame(sk, msg, len);
1721ac714949SMarcel Holtmann 		goto done;
17220381101fSJohan Hedberg 	default:
1723801c1e8dSJohan Hedberg 		mutex_lock(&mgmt_chan_list_lock);
1724801c1e8dSJohan Hedberg 		chan = __hci_mgmt_chan_find(hci_pi(sk)->channel);
1725801c1e8dSJohan Hedberg 		if (chan)
1726fa4335d7SJohan Hedberg 			err = hci_mgmt_cmd(chan, sk, msg, len);
1727801c1e8dSJohan Hedberg 		else
17280381101fSJohan Hedberg 			err = -EINVAL;
1729801c1e8dSJohan Hedberg 
1730801c1e8dSJohan Hedberg 		mutex_unlock(&mgmt_chan_list_lock);
17310381101fSJohan Hedberg 		goto done;
17320381101fSJohan Hedberg 	}
17330381101fSJohan Hedberg 
173470f23020SAndrei Emeltchenko 	hdev = hci_pi(sk)->hdev;
173570f23020SAndrei Emeltchenko 	if (!hdev) {
17361da177e4SLinus Torvalds 		err = -EBADFD;
17371da177e4SLinus Torvalds 		goto done;
17381da177e4SLinus Torvalds 	}
17391da177e4SLinus Torvalds 
17407e21addcSMarcel Holtmann 	if (!test_bit(HCI_UP, &hdev->flags)) {
17417e21addcSMarcel Holtmann 		err = -ENETDOWN;
17427e21addcSMarcel Holtmann 		goto done;
17437e21addcSMarcel Holtmann 	}
17447e21addcSMarcel Holtmann 
174570f23020SAndrei Emeltchenko 	skb = bt_skb_send_alloc(sk, len, msg->msg_flags & MSG_DONTWAIT, &err);
174670f23020SAndrei Emeltchenko 	if (!skb)
17471da177e4SLinus Torvalds 		goto done;
17481da177e4SLinus Torvalds 
17496ce8e9ceSAl Viro 	if (memcpy_from_msg(skb_put(skb, len), msg, len)) {
17501da177e4SLinus Torvalds 		err = -EFAULT;
17511da177e4SLinus Torvalds 		goto drop;
17521da177e4SLinus Torvalds 	}
17531da177e4SLinus Torvalds 
17548528d3f7SMarcel Holtmann 	hci_skb_pkt_type(skb) = skb->data[0];
17551da177e4SLinus Torvalds 	skb_pull(skb, 1);
17561da177e4SLinus Torvalds 
17571bc5ad16SMarcel Holtmann 	if (hci_pi(sk)->channel == HCI_CHANNEL_USER) {
17581bc5ad16SMarcel Holtmann 		/* No permission check is needed for user channel
17591bc5ad16SMarcel Holtmann 		 * since that gets enforced when binding the socket.
17601bc5ad16SMarcel Holtmann 		 *
17611bc5ad16SMarcel Holtmann 		 * However check that the packet type is valid.
17621bc5ad16SMarcel Holtmann 		 */
1763d79f34e3SMarcel Holtmann 		if (hci_skb_pkt_type(skb) != HCI_COMMAND_PKT &&
1764d79f34e3SMarcel Holtmann 		    hci_skb_pkt_type(skb) != HCI_ACLDATA_PKT &&
1765d79f34e3SMarcel Holtmann 		    hci_skb_pkt_type(skb) != HCI_SCODATA_PKT) {
17661bc5ad16SMarcel Holtmann 			err = -EINVAL;
17671bc5ad16SMarcel Holtmann 			goto drop;
17681bc5ad16SMarcel Holtmann 		}
17691bc5ad16SMarcel Holtmann 
17701bc5ad16SMarcel Holtmann 		skb_queue_tail(&hdev->raw_q, skb);
17711bc5ad16SMarcel Holtmann 		queue_work(hdev->workqueue, &hdev->tx_work);
1772d79f34e3SMarcel Holtmann 	} else if (hci_skb_pkt_type(skb) == HCI_COMMAND_PKT) {
177383985319SHarvey Harrison 		u16 opcode = get_unaligned_le16(skb->data);
17741da177e4SLinus Torvalds 		u16 ogf = hci_opcode_ogf(opcode);
17751da177e4SLinus Torvalds 		u16 ocf = hci_opcode_ocf(opcode);
17761da177e4SLinus Torvalds 
17771da177e4SLinus Torvalds 		if (((ogf > HCI_SFLT_MAX_OGF) ||
17783bb3c755SGustavo Padovan 		     !hci_test_bit(ocf & HCI_FLT_OCF_BITS,
17793bb3c755SGustavo Padovan 				   &hci_sec_filter.ocf_mask[ogf])) &&
17801da177e4SLinus Torvalds 		    !capable(CAP_NET_RAW)) {
17811da177e4SLinus Torvalds 			err = -EPERM;
17821da177e4SLinus Torvalds 			goto drop;
17831da177e4SLinus Torvalds 		}
17841da177e4SLinus Torvalds 
17851982162bSMarcel Holtmann 		/* Since the opcode has already been extracted here, store
17861982162bSMarcel Holtmann 		 * a copy of the value for later use by the drivers.
17871982162bSMarcel Holtmann 		 */
17881982162bSMarcel Holtmann 		hci_skb_opcode(skb) = opcode;
17891982162bSMarcel Holtmann 
1790fee746b0SMarcel Holtmann 		if (ogf == 0x3f) {
17911da177e4SLinus Torvalds 			skb_queue_tail(&hdev->raw_q, skb);
17923eff45eaSGustavo F. Padovan 			queue_work(hdev->workqueue, &hdev->tx_work);
17931da177e4SLinus Torvalds 		} else {
179449c922bbSStephen Hemminger 			/* Stand-alone HCI commands must be flagged as
179511714b3dSJohan Hedberg 			 * single-command requests.
179611714b3dSJohan Hedberg 			 */
179744d27137SJohan Hedberg 			bt_cb(skb)->hci.req_flags |= HCI_REQ_START;
179811714b3dSJohan Hedberg 
17991da177e4SLinus Torvalds 			skb_queue_tail(&hdev->cmd_q, skb);
1800c347b765SGustavo F. Padovan 			queue_work(hdev->workqueue, &hdev->cmd_work);
18011da177e4SLinus Torvalds 		}
18021da177e4SLinus Torvalds 	} else {
18031da177e4SLinus Torvalds 		if (!capable(CAP_NET_RAW)) {
18041da177e4SLinus Torvalds 			err = -EPERM;
18051da177e4SLinus Torvalds 			goto drop;
18061da177e4SLinus Torvalds 		}
18071da177e4SLinus Torvalds 
1808d79f34e3SMarcel Holtmann 		if (hci_skb_pkt_type(skb) != HCI_ACLDATA_PKT &&
1809d79f34e3SMarcel Holtmann 		    hci_skb_pkt_type(skb) != HCI_SCODATA_PKT) {
1810bb77543eSMarcel Holtmann 			err = -EINVAL;
1811bb77543eSMarcel Holtmann 			goto drop;
1812bb77543eSMarcel Holtmann 		}
1813bb77543eSMarcel Holtmann 
18141da177e4SLinus Torvalds 		skb_queue_tail(&hdev->raw_q, skb);
18153eff45eaSGustavo F. Padovan 		queue_work(hdev->workqueue, &hdev->tx_work);
18161da177e4SLinus Torvalds 	}
18171da177e4SLinus Torvalds 
18181da177e4SLinus Torvalds 	err = len;
18191da177e4SLinus Torvalds 
18201da177e4SLinus Torvalds done:
18211da177e4SLinus Torvalds 	release_sock(sk);
18221da177e4SLinus Torvalds 	return err;
18231da177e4SLinus Torvalds 
18241da177e4SLinus Torvalds drop:
18251da177e4SLinus Torvalds 	kfree_skb(skb);
18261da177e4SLinus Torvalds 	goto done;
18271da177e4SLinus Torvalds }
18281da177e4SLinus Torvalds 
18298fc9ced3SGustavo Padovan static int hci_sock_setsockopt(struct socket *sock, int level, int optname,
18308fc9ced3SGustavo Padovan 			       char __user *optval, unsigned int len)
18311da177e4SLinus Torvalds {
18321da177e4SLinus Torvalds 	struct hci_ufilter uf = { .opcode = 0 };
18331da177e4SLinus Torvalds 	struct sock *sk = sock->sk;
18341da177e4SLinus Torvalds 	int err = 0, opt = 0;
18351da177e4SLinus Torvalds 
18361da177e4SLinus Torvalds 	BT_DBG("sk %p, opt %d", sk, optname);
18371da177e4SLinus Torvalds 
183847b0f573SMarcel Holtmann 	if (level != SOL_HCI)
183947b0f573SMarcel Holtmann 		return -ENOPROTOOPT;
184047b0f573SMarcel Holtmann 
18411da177e4SLinus Torvalds 	lock_sock(sk);
18421da177e4SLinus Torvalds 
18432f39cdb7SMarcel Holtmann 	if (hci_pi(sk)->channel != HCI_CHANNEL_RAW) {
1844c2371e80SMarcel Holtmann 		err = -EBADFD;
18452f39cdb7SMarcel Holtmann 		goto done;
18462f39cdb7SMarcel Holtmann 	}
18472f39cdb7SMarcel Holtmann 
18481da177e4SLinus Torvalds 	switch (optname) {
18491da177e4SLinus Torvalds 	case HCI_DATA_DIR:
18501da177e4SLinus Torvalds 		if (get_user(opt, (int __user *)optval)) {
18511da177e4SLinus Torvalds 			err = -EFAULT;
18521da177e4SLinus Torvalds 			break;
18531da177e4SLinus Torvalds 		}
18541da177e4SLinus Torvalds 
18551da177e4SLinus Torvalds 		if (opt)
18561da177e4SLinus Torvalds 			hci_pi(sk)->cmsg_mask |= HCI_CMSG_DIR;
18571da177e4SLinus Torvalds 		else
18581da177e4SLinus Torvalds 			hci_pi(sk)->cmsg_mask &= ~HCI_CMSG_DIR;
18591da177e4SLinus Torvalds 		break;
18601da177e4SLinus Torvalds 
18611da177e4SLinus Torvalds 	case HCI_TIME_STAMP:
18621da177e4SLinus Torvalds 		if (get_user(opt, (int __user *)optval)) {
18631da177e4SLinus Torvalds 			err = -EFAULT;
18641da177e4SLinus Torvalds 			break;
18651da177e4SLinus Torvalds 		}
18661da177e4SLinus Torvalds 
18671da177e4SLinus Torvalds 		if (opt)
18681da177e4SLinus Torvalds 			hci_pi(sk)->cmsg_mask |= HCI_CMSG_TSTAMP;
18691da177e4SLinus Torvalds 		else
18701da177e4SLinus Torvalds 			hci_pi(sk)->cmsg_mask &= ~HCI_CMSG_TSTAMP;
18711da177e4SLinus Torvalds 		break;
18721da177e4SLinus Torvalds 
18731da177e4SLinus Torvalds 	case HCI_FILTER:
18740878b666SMarcel Holtmann 		{
18750878b666SMarcel Holtmann 			struct hci_filter *f = &hci_pi(sk)->filter;
18760878b666SMarcel Holtmann 
18770878b666SMarcel Holtmann 			uf.type_mask = f->type_mask;
18780878b666SMarcel Holtmann 			uf.opcode    = f->opcode;
18790878b666SMarcel Holtmann 			uf.event_mask[0] = *((u32 *) f->event_mask + 0);
18800878b666SMarcel Holtmann 			uf.event_mask[1] = *((u32 *) f->event_mask + 1);
18810878b666SMarcel Holtmann 		}
18820878b666SMarcel Holtmann 
18831da177e4SLinus Torvalds 		len = min_t(unsigned int, len, sizeof(uf));
18841da177e4SLinus Torvalds 		if (copy_from_user(&uf, optval, len)) {
18851da177e4SLinus Torvalds 			err = -EFAULT;
18861da177e4SLinus Torvalds 			break;
18871da177e4SLinus Torvalds 		}
18881da177e4SLinus Torvalds 
18891da177e4SLinus Torvalds 		if (!capable(CAP_NET_RAW)) {
18901da177e4SLinus Torvalds 			uf.type_mask &= hci_sec_filter.type_mask;
18911da177e4SLinus Torvalds 			uf.event_mask[0] &= *((u32 *) hci_sec_filter.event_mask + 0);
18921da177e4SLinus Torvalds 			uf.event_mask[1] &= *((u32 *) hci_sec_filter.event_mask + 1);
18931da177e4SLinus Torvalds 		}
18941da177e4SLinus Torvalds 
18951da177e4SLinus Torvalds 		{
18961da177e4SLinus Torvalds 			struct hci_filter *f = &hci_pi(sk)->filter;
18971da177e4SLinus Torvalds 
18981da177e4SLinus Torvalds 			f->type_mask = uf.type_mask;
18991da177e4SLinus Torvalds 			f->opcode    = uf.opcode;
19001da177e4SLinus Torvalds 			*((u32 *) f->event_mask + 0) = uf.event_mask[0];
19011da177e4SLinus Torvalds 			*((u32 *) f->event_mask + 1) = uf.event_mask[1];
19021da177e4SLinus Torvalds 		}
19031da177e4SLinus Torvalds 		break;
19041da177e4SLinus Torvalds 
19051da177e4SLinus Torvalds 	default:
19061da177e4SLinus Torvalds 		err = -ENOPROTOOPT;
19071da177e4SLinus Torvalds 		break;
19081da177e4SLinus Torvalds 	}
19091da177e4SLinus Torvalds 
19102f39cdb7SMarcel Holtmann done:
19111da177e4SLinus Torvalds 	release_sock(sk);
19121da177e4SLinus Torvalds 	return err;
19131da177e4SLinus Torvalds }
19141da177e4SLinus Torvalds 
19158fc9ced3SGustavo Padovan static int hci_sock_getsockopt(struct socket *sock, int level, int optname,
19168fc9ced3SGustavo Padovan 			       char __user *optval, int __user *optlen)
19171da177e4SLinus Torvalds {
19181da177e4SLinus Torvalds 	struct hci_ufilter uf;
19191da177e4SLinus Torvalds 	struct sock *sk = sock->sk;
1920cedc5469SMarcel Holtmann 	int len, opt, err = 0;
1921cedc5469SMarcel Holtmann 
1922cedc5469SMarcel Holtmann 	BT_DBG("sk %p, opt %d", sk, optname);
19231da177e4SLinus Torvalds 
192447b0f573SMarcel Holtmann 	if (level != SOL_HCI)
192547b0f573SMarcel Holtmann 		return -ENOPROTOOPT;
192647b0f573SMarcel Holtmann 
19271da177e4SLinus Torvalds 	if (get_user(len, optlen))
19281da177e4SLinus Torvalds 		return -EFAULT;
19291da177e4SLinus Torvalds 
1930cedc5469SMarcel Holtmann 	lock_sock(sk);
1931cedc5469SMarcel Holtmann 
1932cedc5469SMarcel Holtmann 	if (hci_pi(sk)->channel != HCI_CHANNEL_RAW) {
1933c2371e80SMarcel Holtmann 		err = -EBADFD;
1934cedc5469SMarcel Holtmann 		goto done;
1935cedc5469SMarcel Holtmann 	}
1936cedc5469SMarcel Holtmann 
19371da177e4SLinus Torvalds 	switch (optname) {
19381da177e4SLinus Torvalds 	case HCI_DATA_DIR:
19391da177e4SLinus Torvalds 		if (hci_pi(sk)->cmsg_mask & HCI_CMSG_DIR)
19401da177e4SLinus Torvalds 			opt = 1;
19411da177e4SLinus Torvalds 		else
19421da177e4SLinus Torvalds 			opt = 0;
19431da177e4SLinus Torvalds 
19441da177e4SLinus Torvalds 		if (put_user(opt, optval))
1945cedc5469SMarcel Holtmann 			err = -EFAULT;
19461da177e4SLinus Torvalds 		break;
19471da177e4SLinus Torvalds 
19481da177e4SLinus Torvalds 	case HCI_TIME_STAMP:
19491da177e4SLinus Torvalds 		if (hci_pi(sk)->cmsg_mask & HCI_CMSG_TSTAMP)
19501da177e4SLinus Torvalds 			opt = 1;
19511da177e4SLinus Torvalds 		else
19521da177e4SLinus Torvalds 			opt = 0;
19531da177e4SLinus Torvalds 
19541da177e4SLinus Torvalds 		if (put_user(opt, optval))
1955cedc5469SMarcel Holtmann 			err = -EFAULT;
19561da177e4SLinus Torvalds 		break;
19571da177e4SLinus Torvalds 
19581da177e4SLinus Torvalds 	case HCI_FILTER:
19591da177e4SLinus Torvalds 		{
19601da177e4SLinus Torvalds 			struct hci_filter *f = &hci_pi(sk)->filter;
19611da177e4SLinus Torvalds 
1962e15ca9a0SMathias Krause 			memset(&uf, 0, sizeof(uf));
19631da177e4SLinus Torvalds 			uf.type_mask = f->type_mask;
19641da177e4SLinus Torvalds 			uf.opcode    = f->opcode;
19651da177e4SLinus Torvalds 			uf.event_mask[0] = *((u32 *) f->event_mask + 0);
19661da177e4SLinus Torvalds 			uf.event_mask[1] = *((u32 *) f->event_mask + 1);
19671da177e4SLinus Torvalds 		}
19681da177e4SLinus Torvalds 
19691da177e4SLinus Torvalds 		len = min_t(unsigned int, len, sizeof(uf));
19701da177e4SLinus Torvalds 		if (copy_to_user(optval, &uf, len))
1971cedc5469SMarcel Holtmann 			err = -EFAULT;
19721da177e4SLinus Torvalds 		break;
19731da177e4SLinus Torvalds 
19741da177e4SLinus Torvalds 	default:
1975cedc5469SMarcel Holtmann 		err = -ENOPROTOOPT;
19761da177e4SLinus Torvalds 		break;
19771da177e4SLinus Torvalds 	}
19781da177e4SLinus Torvalds 
1979cedc5469SMarcel Holtmann done:
1980cedc5469SMarcel Holtmann 	release_sock(sk);
1981cedc5469SMarcel Holtmann 	return err;
19821da177e4SLinus Torvalds }
19831da177e4SLinus Torvalds 
198490ddc4f0SEric Dumazet static const struct proto_ops hci_sock_ops = {
19851da177e4SLinus Torvalds 	.family		= PF_BLUETOOTH,
19861da177e4SLinus Torvalds 	.owner		= THIS_MODULE,
19871da177e4SLinus Torvalds 	.release	= hci_sock_release,
19881da177e4SLinus Torvalds 	.bind		= hci_sock_bind,
19891da177e4SLinus Torvalds 	.getname	= hci_sock_getname,
19901da177e4SLinus Torvalds 	.sendmsg	= hci_sock_sendmsg,
19911da177e4SLinus Torvalds 	.recvmsg	= hci_sock_recvmsg,
19921da177e4SLinus Torvalds 	.ioctl		= hci_sock_ioctl,
1993*7a6038b3SArnd Bergmann #ifdef CONFIG_COMPAT
1994*7a6038b3SArnd Bergmann 	.compat_ioctl	= hci_sock_compat_ioctl,
1995*7a6038b3SArnd Bergmann #endif
1996a11e1d43SLinus Torvalds 	.poll		= datagram_poll,
19971da177e4SLinus Torvalds 	.listen		= sock_no_listen,
19981da177e4SLinus Torvalds 	.shutdown	= sock_no_shutdown,
19991da177e4SLinus Torvalds 	.setsockopt	= hci_sock_setsockopt,
20001da177e4SLinus Torvalds 	.getsockopt	= hci_sock_getsockopt,
20011da177e4SLinus Torvalds 	.connect	= sock_no_connect,
20021da177e4SLinus Torvalds 	.socketpair	= sock_no_socketpair,
20031da177e4SLinus Torvalds 	.accept		= sock_no_accept,
20041da177e4SLinus Torvalds 	.mmap		= sock_no_mmap
20051da177e4SLinus Torvalds };
20061da177e4SLinus Torvalds 
20071da177e4SLinus Torvalds static struct proto hci_sk_proto = {
20081da177e4SLinus Torvalds 	.name		= "HCI",
20091da177e4SLinus Torvalds 	.owner		= THIS_MODULE,
20101da177e4SLinus Torvalds 	.obj_size	= sizeof(struct hci_pinfo)
20111da177e4SLinus Torvalds };
20121da177e4SLinus Torvalds 
20133f378b68SEric Paris static int hci_sock_create(struct net *net, struct socket *sock, int protocol,
20143f378b68SEric Paris 			   int kern)
20151da177e4SLinus Torvalds {
20161da177e4SLinus Torvalds 	struct sock *sk;
20171da177e4SLinus Torvalds 
20181da177e4SLinus Torvalds 	BT_DBG("sock %p", sock);
20191da177e4SLinus Torvalds 
20201da177e4SLinus Torvalds 	if (sock->type != SOCK_RAW)
20211da177e4SLinus Torvalds 		return -ESOCKTNOSUPPORT;
20221da177e4SLinus Torvalds 
20231da177e4SLinus Torvalds 	sock->ops = &hci_sock_ops;
20241da177e4SLinus Torvalds 
202511aa9c28SEric W. Biederman 	sk = sk_alloc(net, PF_BLUETOOTH, GFP_ATOMIC, &hci_sk_proto, kern);
20261da177e4SLinus Torvalds 	if (!sk)
20271da177e4SLinus Torvalds 		return -ENOMEM;
20281da177e4SLinus Torvalds 
20291da177e4SLinus Torvalds 	sock_init_data(sock, sk);
20301da177e4SLinus Torvalds 
20311da177e4SLinus Torvalds 	sock_reset_flag(sk, SOCK_ZAPPED);
20321da177e4SLinus Torvalds 
20331da177e4SLinus Torvalds 	sk->sk_protocol = protocol;
20341da177e4SLinus Torvalds 
20351da177e4SLinus Torvalds 	sock->state = SS_UNCONNECTED;
20361da177e4SLinus Torvalds 	sk->sk_state = BT_OPEN;
20371da177e4SLinus Torvalds 
20381da177e4SLinus Torvalds 	bt_sock_link(&hci_sk_list, sk);
20391da177e4SLinus Torvalds 	return 0;
20401da177e4SLinus Torvalds }
20411da177e4SLinus Torvalds 
2042ec1b4cf7SStephen Hemminger static const struct net_proto_family hci_sock_family_ops = {
20431da177e4SLinus Torvalds 	.family	= PF_BLUETOOTH,
20441da177e4SLinus Torvalds 	.owner	= THIS_MODULE,
20451da177e4SLinus Torvalds 	.create	= hci_sock_create,
20461da177e4SLinus Torvalds };
20471da177e4SLinus Torvalds 
20481da177e4SLinus Torvalds int __init hci_sock_init(void)
20491da177e4SLinus Torvalds {
20501da177e4SLinus Torvalds 	int err;
20511da177e4SLinus Torvalds 
2052b0a8e282SMarcel Holtmann 	BUILD_BUG_ON(sizeof(struct sockaddr_hci) > sizeof(struct sockaddr));
2053b0a8e282SMarcel Holtmann 
20541da177e4SLinus Torvalds 	err = proto_register(&hci_sk_proto, 0);
20551da177e4SLinus Torvalds 	if (err < 0)
20561da177e4SLinus Torvalds 		return err;
20571da177e4SLinus Torvalds 
20581da177e4SLinus Torvalds 	err = bt_sock_register(BTPROTO_HCI, &hci_sock_family_ops);
2059f7c86637SMasatake YAMATO 	if (err < 0) {
2060f7c86637SMasatake YAMATO 		BT_ERR("HCI socket registration failed");
20611da177e4SLinus Torvalds 		goto error;
2062f7c86637SMasatake YAMATO 	}
2063f7c86637SMasatake YAMATO 
2064b0316615SAl Viro 	err = bt_procfs_init(&init_net, "hci", &hci_sk_list, NULL);
2065f7c86637SMasatake YAMATO 	if (err < 0) {
2066f7c86637SMasatake YAMATO 		BT_ERR("Failed to create HCI proc file");
2067f7c86637SMasatake YAMATO 		bt_sock_unregister(BTPROTO_HCI);
2068f7c86637SMasatake YAMATO 		goto error;
2069f7c86637SMasatake YAMATO 	}
20701da177e4SLinus Torvalds 
20711da177e4SLinus Torvalds 	BT_INFO("HCI socket layer initialized");
20721da177e4SLinus Torvalds 
20731da177e4SLinus Torvalds 	return 0;
20741da177e4SLinus Torvalds 
20751da177e4SLinus Torvalds error:
20761da177e4SLinus Torvalds 	proto_unregister(&hci_sk_proto);
20771da177e4SLinus Torvalds 	return err;
20781da177e4SLinus Torvalds }
20791da177e4SLinus Torvalds 
2080b7440a14SAnand Gadiyar void hci_sock_cleanup(void)
20811da177e4SLinus Torvalds {
2082f7c86637SMasatake YAMATO 	bt_procfs_cleanup(&init_net, "hci");
20835e9d7f86SDavid Herrmann 	bt_sock_unregister(BTPROTO_HCI);
20841da177e4SLinus Torvalds 	proto_unregister(&hci_sk_proto);
20851da177e4SLinus Torvalds }
2086