xref: /openbmc/linux/net/bluetooth/hci_sock.c (revision d0bef1d26fb6fdad818f3d15a178d51e2a8478ae)
11da177e4SLinus Torvalds /*
21da177e4SLinus Torvalds    BlueZ - Bluetooth protocol stack for Linux
31da177e4SLinus Torvalds    Copyright (C) 2000-2001 Qualcomm Incorporated
41da177e4SLinus Torvalds 
51da177e4SLinus Torvalds    Written 2000,2001 by Maxim Krasnyansky <maxk@qualcomm.com>
61da177e4SLinus Torvalds 
71da177e4SLinus Torvalds    This program is free software; you can redistribute it and/or modify
81da177e4SLinus Torvalds    it under the terms of the GNU General Public License version 2 as
91da177e4SLinus Torvalds    published by the Free Software Foundation;
101da177e4SLinus Torvalds 
111da177e4SLinus Torvalds    THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
121da177e4SLinus Torvalds    OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
131da177e4SLinus Torvalds    FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT OF THIRD PARTY RIGHTS.
141da177e4SLinus Torvalds    IN NO EVENT SHALL THE COPYRIGHT HOLDER(S) AND AUTHOR(S) BE LIABLE FOR ANY
151da177e4SLinus Torvalds    CLAIM, OR ANY SPECIAL INDIRECT OR CONSEQUENTIAL DAMAGES, OR ANY DAMAGES
161da177e4SLinus Torvalds    WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
171da177e4SLinus Torvalds    ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
181da177e4SLinus Torvalds    OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
191da177e4SLinus Torvalds 
201da177e4SLinus Torvalds    ALL LIABILITY, INCLUDING LIABILITY FOR INFRINGEMENT OF ANY PATENTS,
211da177e4SLinus Torvalds    COPYRIGHTS, TRADEMARKS OR OTHER RIGHTS, RELATING TO USE OF THIS
221da177e4SLinus Torvalds    SOFTWARE IS DISCLAIMED.
231da177e4SLinus Torvalds */
241da177e4SLinus Torvalds 
251da177e4SLinus Torvalds /* Bluetooth HCI sockets. */
261da177e4SLinus Torvalds 
278c520a59SGustavo Padovan #include <linux/export.h>
28787b306cSJohannes Berg #include <linux/utsname.h>
2970ecce91SMarcel Holtmann #include <linux/sched.h>
301da177e4SLinus Torvalds #include <asm/unaligned.h>
311da177e4SLinus Torvalds 
321da177e4SLinus Torvalds #include <net/bluetooth/bluetooth.h>
331da177e4SLinus Torvalds #include <net/bluetooth/hci_core.h>
34cd82e61cSMarcel Holtmann #include <net/bluetooth/hci_mon.h>
35fa4335d7SJohan Hedberg #include <net/bluetooth/mgmt.h>
36fa4335d7SJohan Hedberg 
37fa4335d7SJohan Hedberg #include "mgmt_util.h"
381da177e4SLinus Torvalds 
39801c1e8dSJohan Hedberg static LIST_HEAD(mgmt_chan_list);
40801c1e8dSJohan Hedberg static DEFINE_MUTEX(mgmt_chan_list_lock);
41801c1e8dSJohan Hedberg 
4270ecce91SMarcel Holtmann static DEFINE_IDA(sock_cookie_ida);
4370ecce91SMarcel Holtmann 
44cd82e61cSMarcel Holtmann static atomic_t monitor_promisc = ATOMIC_INIT(0);
45cd82e61cSMarcel Holtmann 
461da177e4SLinus Torvalds /* ----- HCI socket interface ----- */
471da177e4SLinus Torvalds 
48863def58SMarcel Holtmann /* Socket info */
49863def58SMarcel Holtmann #define hci_pi(sk) ((struct hci_pinfo *) sk)
50863def58SMarcel Holtmann 
51863def58SMarcel Holtmann struct hci_pinfo {
52863def58SMarcel Holtmann 	struct bt_sock    bt;
53863def58SMarcel Holtmann 	struct hci_dev    *hdev;
54863def58SMarcel Holtmann 	struct hci_filter filter;
55863def58SMarcel Holtmann 	__u32             cmsg_mask;
56863def58SMarcel Holtmann 	unsigned short    channel;
576befc644SMarcel Holtmann 	unsigned long     flags;
5870ecce91SMarcel Holtmann 	__u32             cookie;
5970ecce91SMarcel Holtmann 	char              comm[TASK_COMM_LEN];
60863def58SMarcel Holtmann };
61863def58SMarcel Holtmann 
626befc644SMarcel Holtmann void hci_sock_set_flag(struct sock *sk, int nr)
636befc644SMarcel Holtmann {
646befc644SMarcel Holtmann 	set_bit(nr, &hci_pi(sk)->flags);
656befc644SMarcel Holtmann }
666befc644SMarcel Holtmann 
676befc644SMarcel Holtmann void hci_sock_clear_flag(struct sock *sk, int nr)
686befc644SMarcel Holtmann {
696befc644SMarcel Holtmann 	clear_bit(nr, &hci_pi(sk)->flags);
706befc644SMarcel Holtmann }
716befc644SMarcel Holtmann 
72c85be545SMarcel Holtmann int hci_sock_test_flag(struct sock *sk, int nr)
73c85be545SMarcel Holtmann {
74c85be545SMarcel Holtmann 	return test_bit(nr, &hci_pi(sk)->flags);
75c85be545SMarcel Holtmann }
76c85be545SMarcel Holtmann 
77d0f172b1SJohan Hedberg unsigned short hci_sock_get_channel(struct sock *sk)
78d0f172b1SJohan Hedberg {
79d0f172b1SJohan Hedberg 	return hci_pi(sk)->channel;
80d0f172b1SJohan Hedberg }
81d0f172b1SJohan Hedberg 
8270ecce91SMarcel Holtmann u32 hci_sock_get_cookie(struct sock *sk)
8370ecce91SMarcel Holtmann {
8470ecce91SMarcel Holtmann 	return hci_pi(sk)->cookie;
8570ecce91SMarcel Holtmann }
8670ecce91SMarcel Holtmann 
87df1cb87aSMarcel Holtmann static bool hci_sock_gen_cookie(struct sock *sk)
88df1cb87aSMarcel Holtmann {
89df1cb87aSMarcel Holtmann 	int id = hci_pi(sk)->cookie;
90df1cb87aSMarcel Holtmann 
91df1cb87aSMarcel Holtmann 	if (!id) {
92df1cb87aSMarcel Holtmann 		id = ida_simple_get(&sock_cookie_ida, 1, 0, GFP_KERNEL);
93df1cb87aSMarcel Holtmann 		if (id < 0)
94df1cb87aSMarcel Holtmann 			id = 0xffffffff;
95df1cb87aSMarcel Holtmann 
96df1cb87aSMarcel Holtmann 		hci_pi(sk)->cookie = id;
97df1cb87aSMarcel Holtmann 		get_task_comm(hci_pi(sk)->comm, current);
98df1cb87aSMarcel Holtmann 		return true;
99df1cb87aSMarcel Holtmann 	}
100df1cb87aSMarcel Holtmann 
101df1cb87aSMarcel Holtmann 	return false;
102df1cb87aSMarcel Holtmann }
103df1cb87aSMarcel Holtmann 
104df1cb87aSMarcel Holtmann static void hci_sock_free_cookie(struct sock *sk)
105df1cb87aSMarcel Holtmann {
106df1cb87aSMarcel Holtmann 	int id = hci_pi(sk)->cookie;
107df1cb87aSMarcel Holtmann 
108df1cb87aSMarcel Holtmann 	if (id) {
109df1cb87aSMarcel Holtmann 		hci_pi(sk)->cookie = 0xffffffff;
110df1cb87aSMarcel Holtmann 		ida_simple_remove(&sock_cookie_ida, id);
111df1cb87aSMarcel Holtmann 	}
112df1cb87aSMarcel Holtmann }
113df1cb87aSMarcel Holtmann 
1149391976aSJiri Slaby static inline int hci_test_bit(int nr, const void *addr)
1151da177e4SLinus Torvalds {
1169391976aSJiri Slaby 	return *((const __u32 *) addr + (nr >> 5)) & ((__u32) 1 << (nr & 31));
1171da177e4SLinus Torvalds }
1181da177e4SLinus Torvalds 
1191da177e4SLinus Torvalds /* Security filter */
1203ad254f7SMarcel Holtmann #define HCI_SFLT_MAX_OGF  5
1213ad254f7SMarcel Holtmann 
1223ad254f7SMarcel Holtmann struct hci_sec_filter {
1233ad254f7SMarcel Holtmann 	__u32 type_mask;
1243ad254f7SMarcel Holtmann 	__u32 event_mask[2];
1253ad254f7SMarcel Holtmann 	__u32 ocf_mask[HCI_SFLT_MAX_OGF + 1][4];
1263ad254f7SMarcel Holtmann };
1273ad254f7SMarcel Holtmann 
1287e67c112SMarcel Holtmann static const struct hci_sec_filter hci_sec_filter = {
1291da177e4SLinus Torvalds 	/* Packet types */
1301da177e4SLinus Torvalds 	0x10,
1311da177e4SLinus Torvalds 	/* Events */
132dd7f5527SMarcel Holtmann 	{ 0x1000d9fe, 0x0000b00c },
1331da177e4SLinus Torvalds 	/* Commands */
1341da177e4SLinus Torvalds 	{
1351da177e4SLinus Torvalds 		{ 0x0 },
1361da177e4SLinus Torvalds 		/* OGF_LINK_CTL */
1377c631a67SMarcel Holtmann 		{ 0xbe000006, 0x00000001, 0x00000000, 0x00 },
1381da177e4SLinus Torvalds 		/* OGF_LINK_POLICY */
1397c631a67SMarcel Holtmann 		{ 0x00005200, 0x00000000, 0x00000000, 0x00 },
1401da177e4SLinus Torvalds 		/* OGF_HOST_CTL */
1417c631a67SMarcel Holtmann 		{ 0xaab00200, 0x2b402aaa, 0x05220154, 0x00 },
1421da177e4SLinus Torvalds 		/* OGF_INFO_PARAM */
1437c631a67SMarcel Holtmann 		{ 0x000002be, 0x00000000, 0x00000000, 0x00 },
1441da177e4SLinus Torvalds 		/* OGF_STATUS_PARAM */
1457c631a67SMarcel Holtmann 		{ 0x000000ea, 0x00000000, 0x00000000, 0x00 }
1461da177e4SLinus Torvalds 	}
1471da177e4SLinus Torvalds };
1481da177e4SLinus Torvalds 
1491da177e4SLinus Torvalds static struct bt_sock_list hci_sk_list = {
150d5fb2962SRobert P. J. Day 	.lock = __RW_LOCK_UNLOCKED(hci_sk_list.lock)
1511da177e4SLinus Torvalds };
1521da177e4SLinus Torvalds 
153f81fe64fSMarcel Holtmann static bool is_filtered_packet(struct sock *sk, struct sk_buff *skb)
154f81fe64fSMarcel Holtmann {
155f81fe64fSMarcel Holtmann 	struct hci_filter *flt;
156f81fe64fSMarcel Holtmann 	int flt_type, flt_event;
157f81fe64fSMarcel Holtmann 
158f81fe64fSMarcel Holtmann 	/* Apply filter */
159f81fe64fSMarcel Holtmann 	flt = &hci_pi(sk)->filter;
160f81fe64fSMarcel Holtmann 
161d79f34e3SMarcel Holtmann 	flt_type = hci_skb_pkt_type(skb) & HCI_FLT_TYPE_BITS;
162f81fe64fSMarcel Holtmann 
163f81fe64fSMarcel Holtmann 	if (!test_bit(flt_type, &flt->type_mask))
164f81fe64fSMarcel Holtmann 		return true;
165f81fe64fSMarcel Holtmann 
166f81fe64fSMarcel Holtmann 	/* Extra filter for event packets only */
167d79f34e3SMarcel Holtmann 	if (hci_skb_pkt_type(skb) != HCI_EVENT_PKT)
168f81fe64fSMarcel Holtmann 		return false;
169f81fe64fSMarcel Holtmann 
170f81fe64fSMarcel Holtmann 	flt_event = (*(__u8 *)skb->data & HCI_FLT_EVENT_BITS);
171f81fe64fSMarcel Holtmann 
172f81fe64fSMarcel Holtmann 	if (!hci_test_bit(flt_event, &flt->event_mask))
173f81fe64fSMarcel Holtmann 		return true;
174f81fe64fSMarcel Holtmann 
175f81fe64fSMarcel Holtmann 	/* Check filter only when opcode is set */
176f81fe64fSMarcel Holtmann 	if (!flt->opcode)
177f81fe64fSMarcel Holtmann 		return false;
178f81fe64fSMarcel Holtmann 
179f81fe64fSMarcel Holtmann 	if (flt_event == HCI_EV_CMD_COMPLETE &&
180f81fe64fSMarcel Holtmann 	    flt->opcode != get_unaligned((__le16 *)(skb->data + 3)))
181f81fe64fSMarcel Holtmann 		return true;
182f81fe64fSMarcel Holtmann 
183f81fe64fSMarcel Holtmann 	if (flt_event == HCI_EV_CMD_STATUS &&
184f81fe64fSMarcel Holtmann 	    flt->opcode != get_unaligned((__le16 *)(skb->data + 4)))
185f81fe64fSMarcel Holtmann 		return true;
186f81fe64fSMarcel Holtmann 
187f81fe64fSMarcel Holtmann 	return false;
188f81fe64fSMarcel Holtmann }
189f81fe64fSMarcel Holtmann 
1901da177e4SLinus Torvalds /* Send frame to RAW socket */
191470fe1b5SMarcel Holtmann void hci_send_to_sock(struct hci_dev *hdev, struct sk_buff *skb)
1921da177e4SLinus Torvalds {
1931da177e4SLinus Torvalds 	struct sock *sk;
194e0edf373SMarcel Holtmann 	struct sk_buff *skb_copy = NULL;
1951da177e4SLinus Torvalds 
1961da177e4SLinus Torvalds 	BT_DBG("hdev %p len %d", hdev, skb->len);
1971da177e4SLinus Torvalds 
1981da177e4SLinus Torvalds 	read_lock(&hci_sk_list.lock);
199470fe1b5SMarcel Holtmann 
200b67bfe0dSSasha Levin 	sk_for_each(sk, &hci_sk_list.head) {
2011da177e4SLinus Torvalds 		struct sk_buff *nskb;
2021da177e4SLinus Torvalds 
2031da177e4SLinus Torvalds 		if (sk->sk_state != BT_BOUND || hci_pi(sk)->hdev != hdev)
2041da177e4SLinus Torvalds 			continue;
2051da177e4SLinus Torvalds 
2061da177e4SLinus Torvalds 		/* Don't send frame to the socket it came from */
2071da177e4SLinus Torvalds 		if (skb->sk == sk)
2081da177e4SLinus Torvalds 			continue;
2091da177e4SLinus Torvalds 
21023500189SMarcel Holtmann 		if (hci_pi(sk)->channel == HCI_CHANNEL_RAW) {
211d79f34e3SMarcel Holtmann 			if (hci_skb_pkt_type(skb) != HCI_COMMAND_PKT &&
212d79f34e3SMarcel Holtmann 			    hci_skb_pkt_type(skb) != HCI_EVENT_PKT &&
213d79f34e3SMarcel Holtmann 			    hci_skb_pkt_type(skb) != HCI_ACLDATA_PKT &&
214d79f34e3SMarcel Holtmann 			    hci_skb_pkt_type(skb) != HCI_SCODATA_PKT)
215bb77543eSMarcel Holtmann 				continue;
216f81fe64fSMarcel Holtmann 			if (is_filtered_packet(sk, skb))
2171da177e4SLinus Torvalds 				continue;
21823500189SMarcel Holtmann 		} else if (hci_pi(sk)->channel == HCI_CHANNEL_USER) {
21923500189SMarcel Holtmann 			if (!bt_cb(skb)->incoming)
22023500189SMarcel Holtmann 				continue;
221d79f34e3SMarcel Holtmann 			if (hci_skb_pkt_type(skb) != HCI_EVENT_PKT &&
222d79f34e3SMarcel Holtmann 			    hci_skb_pkt_type(skb) != HCI_ACLDATA_PKT &&
223d79f34e3SMarcel Holtmann 			    hci_skb_pkt_type(skb) != HCI_SCODATA_PKT)
22423500189SMarcel Holtmann 				continue;
22523500189SMarcel Holtmann 		} else {
22623500189SMarcel Holtmann 			/* Don't send frame to other channel types */
22723500189SMarcel Holtmann 			continue;
22823500189SMarcel Holtmann 		}
2291da177e4SLinus Torvalds 
230e0edf373SMarcel Holtmann 		if (!skb_copy) {
231e0edf373SMarcel Holtmann 			/* Create a private copy with headroom */
232bad93e9dSOctavian Purdila 			skb_copy = __pskb_copy_fclone(skb, 1, GFP_ATOMIC, true);
233e0edf373SMarcel Holtmann 			if (!skb_copy)
2341da177e4SLinus Torvalds 				continue;
2351da177e4SLinus Torvalds 
2361da177e4SLinus Torvalds 			/* Put type byte before the data */
237d79f34e3SMarcel Holtmann 			memcpy(skb_push(skb_copy, 1), &hci_skb_pkt_type(skb), 1);
238e0edf373SMarcel Holtmann 		}
239e0edf373SMarcel Holtmann 
240e0edf373SMarcel Holtmann 		nskb = skb_clone(skb_copy, GFP_ATOMIC);
241e0edf373SMarcel Holtmann 		if (!nskb)
242e0edf373SMarcel Holtmann 			continue;
2431da177e4SLinus Torvalds 
2441da177e4SLinus Torvalds 		if (sock_queue_rcv_skb(sk, nskb))
2451da177e4SLinus Torvalds 			kfree_skb(nskb);
2461da177e4SLinus Torvalds 	}
247470fe1b5SMarcel Holtmann 
248470fe1b5SMarcel Holtmann 	read_unlock(&hci_sk_list.lock);
249e0edf373SMarcel Holtmann 
250e0edf373SMarcel Holtmann 	kfree_skb(skb_copy);
251470fe1b5SMarcel Holtmann }
252470fe1b5SMarcel Holtmann 
2537129069eSJohan Hedberg /* Send frame to sockets with specific channel */
2547129069eSJohan Hedberg void hci_send_to_channel(unsigned short channel, struct sk_buff *skb,
255c08b1a1dSMarcel Holtmann 			 int flag, struct sock *skip_sk)
256470fe1b5SMarcel Holtmann {
257470fe1b5SMarcel Holtmann 	struct sock *sk;
258470fe1b5SMarcel Holtmann 
2597129069eSJohan Hedberg 	BT_DBG("channel %u len %d", channel, skb->len);
260470fe1b5SMarcel Holtmann 
261470fe1b5SMarcel Holtmann 	read_lock(&hci_sk_list.lock);
262470fe1b5SMarcel Holtmann 
263b67bfe0dSSasha Levin 	sk_for_each(sk, &hci_sk_list.head) {
264470fe1b5SMarcel Holtmann 		struct sk_buff *nskb;
265470fe1b5SMarcel Holtmann 
266c08b1a1dSMarcel Holtmann 		/* Ignore socket without the flag set */
267c85be545SMarcel Holtmann 		if (!hci_sock_test_flag(sk, flag))
268c08b1a1dSMarcel Holtmann 			continue;
269c08b1a1dSMarcel Holtmann 
270470fe1b5SMarcel Holtmann 		/* Skip the original socket */
271470fe1b5SMarcel Holtmann 		if (sk == skip_sk)
272470fe1b5SMarcel Holtmann 			continue;
273470fe1b5SMarcel Holtmann 
274470fe1b5SMarcel Holtmann 		if (sk->sk_state != BT_BOUND)
275470fe1b5SMarcel Holtmann 			continue;
276470fe1b5SMarcel Holtmann 
2777129069eSJohan Hedberg 		if (hci_pi(sk)->channel != channel)
278d7f72f61SMarcel Holtmann 			continue;
279d7f72f61SMarcel Holtmann 
280d7f72f61SMarcel Holtmann 		nskb = skb_clone(skb, GFP_ATOMIC);
281d7f72f61SMarcel Holtmann 		if (!nskb)
282d7f72f61SMarcel Holtmann 			continue;
283d7f72f61SMarcel Holtmann 
284d7f72f61SMarcel Holtmann 		if (sock_queue_rcv_skb(sk, nskb))
285d7f72f61SMarcel Holtmann 			kfree_skb(nskb);
286d7f72f61SMarcel Holtmann 	}
287d7f72f61SMarcel Holtmann 
288d7f72f61SMarcel Holtmann 	read_unlock(&hci_sk_list.lock);
289d7f72f61SMarcel Holtmann }
290d7f72f61SMarcel Holtmann 
291cd82e61cSMarcel Holtmann /* Send frame to monitor socket */
292cd82e61cSMarcel Holtmann void hci_send_to_monitor(struct hci_dev *hdev, struct sk_buff *skb)
293cd82e61cSMarcel Holtmann {
294cd82e61cSMarcel Holtmann 	struct sk_buff *skb_copy = NULL;
2952b531294SMarcel Holtmann 	struct hci_mon_hdr *hdr;
296cd82e61cSMarcel Holtmann 	__le16 opcode;
297cd82e61cSMarcel Holtmann 
298cd82e61cSMarcel Holtmann 	if (!atomic_read(&monitor_promisc))
299cd82e61cSMarcel Holtmann 		return;
300cd82e61cSMarcel Holtmann 
301cd82e61cSMarcel Holtmann 	BT_DBG("hdev %p len %d", hdev, skb->len);
302cd82e61cSMarcel Holtmann 
303d79f34e3SMarcel Holtmann 	switch (hci_skb_pkt_type(skb)) {
304cd82e61cSMarcel Holtmann 	case HCI_COMMAND_PKT:
305dcf4adbfSJoe Perches 		opcode = cpu_to_le16(HCI_MON_COMMAND_PKT);
306cd82e61cSMarcel Holtmann 		break;
307cd82e61cSMarcel Holtmann 	case HCI_EVENT_PKT:
308dcf4adbfSJoe Perches 		opcode = cpu_to_le16(HCI_MON_EVENT_PKT);
309cd82e61cSMarcel Holtmann 		break;
310cd82e61cSMarcel Holtmann 	case HCI_ACLDATA_PKT:
311cd82e61cSMarcel Holtmann 		if (bt_cb(skb)->incoming)
312dcf4adbfSJoe Perches 			opcode = cpu_to_le16(HCI_MON_ACL_RX_PKT);
313cd82e61cSMarcel Holtmann 		else
314dcf4adbfSJoe Perches 			opcode = cpu_to_le16(HCI_MON_ACL_TX_PKT);
315cd82e61cSMarcel Holtmann 		break;
316cd82e61cSMarcel Holtmann 	case HCI_SCODATA_PKT:
317cd82e61cSMarcel Holtmann 		if (bt_cb(skb)->incoming)
318dcf4adbfSJoe Perches 			opcode = cpu_to_le16(HCI_MON_SCO_RX_PKT);
319cd82e61cSMarcel Holtmann 		else
320dcf4adbfSJoe Perches 			opcode = cpu_to_le16(HCI_MON_SCO_TX_PKT);
321cd82e61cSMarcel Holtmann 		break;
322e875ff84SMarcel Holtmann 	case HCI_DIAG_PKT:
323e875ff84SMarcel Holtmann 		opcode = cpu_to_le16(HCI_MON_VENDOR_DIAG);
324e875ff84SMarcel Holtmann 		break;
325cd82e61cSMarcel Holtmann 	default:
326cd82e61cSMarcel Holtmann 		return;
327cd82e61cSMarcel Holtmann 	}
328cd82e61cSMarcel Holtmann 
3292b531294SMarcel Holtmann 	/* Create a private copy with headroom */
3302b531294SMarcel Holtmann 	skb_copy = __pskb_copy_fclone(skb, HCI_MON_HDR_SIZE, GFP_ATOMIC, true);
3312b531294SMarcel Holtmann 	if (!skb_copy)
3322b531294SMarcel Holtmann 		return;
3332b531294SMarcel Holtmann 
3342b531294SMarcel Holtmann 	/* Put header before the data */
3352b531294SMarcel Holtmann 	hdr = (void *)skb_push(skb_copy, HCI_MON_HDR_SIZE);
3362b531294SMarcel Holtmann 	hdr->opcode = opcode;
3372b531294SMarcel Holtmann 	hdr->index = cpu_to_le16(hdev->id);
3382b531294SMarcel Holtmann 	hdr->len = cpu_to_le16(skb->len);
3392b531294SMarcel Holtmann 
340c08b1a1dSMarcel Holtmann 	hci_send_to_channel(HCI_CHANNEL_MONITOR, skb_copy,
341c08b1a1dSMarcel Holtmann 			    HCI_SOCK_TRUSTED, NULL);
342cd82e61cSMarcel Holtmann 	kfree_skb(skb_copy);
343cd82e61cSMarcel Holtmann }
344cd82e61cSMarcel Holtmann 
34538ceaa00SMarcel Holtmann void hci_send_monitor_ctrl_event(struct hci_dev *hdev, u16 event,
34638ceaa00SMarcel Holtmann 				 void *data, u16 data_len, ktime_t tstamp,
34738ceaa00SMarcel Holtmann 				 int flag, struct sock *skip_sk)
34838ceaa00SMarcel Holtmann {
34938ceaa00SMarcel Holtmann 	struct sock *sk;
35038ceaa00SMarcel Holtmann 	__le16 index;
35138ceaa00SMarcel Holtmann 
35238ceaa00SMarcel Holtmann 	if (hdev)
35338ceaa00SMarcel Holtmann 		index = cpu_to_le16(hdev->id);
35438ceaa00SMarcel Holtmann 	else
35538ceaa00SMarcel Holtmann 		index = cpu_to_le16(MGMT_INDEX_NONE);
35638ceaa00SMarcel Holtmann 
35738ceaa00SMarcel Holtmann 	read_lock(&hci_sk_list.lock);
35838ceaa00SMarcel Holtmann 
35938ceaa00SMarcel Holtmann 	sk_for_each(sk, &hci_sk_list.head) {
36038ceaa00SMarcel Holtmann 		struct hci_mon_hdr *hdr;
36138ceaa00SMarcel Holtmann 		struct sk_buff *skb;
36238ceaa00SMarcel Holtmann 
36338ceaa00SMarcel Holtmann 		if (hci_pi(sk)->channel != HCI_CHANNEL_CONTROL)
36438ceaa00SMarcel Holtmann 			continue;
36538ceaa00SMarcel Holtmann 
36638ceaa00SMarcel Holtmann 		/* Ignore socket without the flag set */
36738ceaa00SMarcel Holtmann 		if (!hci_sock_test_flag(sk, flag))
36838ceaa00SMarcel Holtmann 			continue;
36938ceaa00SMarcel Holtmann 
37038ceaa00SMarcel Holtmann 		/* Skip the original socket */
37138ceaa00SMarcel Holtmann 		if (sk == skip_sk)
37238ceaa00SMarcel Holtmann 			continue;
37338ceaa00SMarcel Holtmann 
37438ceaa00SMarcel Holtmann 		skb = bt_skb_alloc(6 + data_len, GFP_ATOMIC);
37538ceaa00SMarcel Holtmann 		if (!skb)
37638ceaa00SMarcel Holtmann 			continue;
37738ceaa00SMarcel Holtmann 
37838ceaa00SMarcel Holtmann 		put_unaligned_le32(hci_pi(sk)->cookie, skb_put(skb, 4));
37938ceaa00SMarcel Holtmann 		put_unaligned_le16(event, skb_put(skb, 2));
38038ceaa00SMarcel Holtmann 
38138ceaa00SMarcel Holtmann 		if (data)
38238ceaa00SMarcel Holtmann 			memcpy(skb_put(skb, data_len), data, data_len);
38338ceaa00SMarcel Holtmann 
38438ceaa00SMarcel Holtmann 		skb->tstamp = tstamp;
38538ceaa00SMarcel Holtmann 
38638ceaa00SMarcel Holtmann 		hdr = (void *)skb_push(skb, HCI_MON_HDR_SIZE);
38738ceaa00SMarcel Holtmann 		hdr->opcode = cpu_to_le16(HCI_MON_CTRL_EVENT);
38838ceaa00SMarcel Holtmann 		hdr->index = index;
38938ceaa00SMarcel Holtmann 		hdr->len = cpu_to_le16(skb->len - HCI_MON_HDR_SIZE);
39038ceaa00SMarcel Holtmann 
39138ceaa00SMarcel Holtmann 		hci_send_to_channel(HCI_CHANNEL_MONITOR, skb,
39238ceaa00SMarcel Holtmann 				    HCI_SOCK_TRUSTED, NULL);
39338ceaa00SMarcel Holtmann 		kfree_skb(skb);
39438ceaa00SMarcel Holtmann 	}
39538ceaa00SMarcel Holtmann 
39638ceaa00SMarcel Holtmann 	read_unlock(&hci_sk_list.lock);
39738ceaa00SMarcel Holtmann }
39838ceaa00SMarcel Holtmann 
399cd82e61cSMarcel Holtmann static struct sk_buff *create_monitor_event(struct hci_dev *hdev, int event)
400cd82e61cSMarcel Holtmann {
401cd82e61cSMarcel Holtmann 	struct hci_mon_hdr *hdr;
402cd82e61cSMarcel Holtmann 	struct hci_mon_new_index *ni;
4036c566dd5SMarcel Holtmann 	struct hci_mon_index_info *ii;
404cd82e61cSMarcel Holtmann 	struct sk_buff *skb;
405cd82e61cSMarcel Holtmann 	__le16 opcode;
406cd82e61cSMarcel Holtmann 
407cd82e61cSMarcel Holtmann 	switch (event) {
408cd82e61cSMarcel Holtmann 	case HCI_DEV_REG:
409cd82e61cSMarcel Holtmann 		skb = bt_skb_alloc(HCI_MON_NEW_INDEX_SIZE, GFP_ATOMIC);
410cd82e61cSMarcel Holtmann 		if (!skb)
411cd82e61cSMarcel Holtmann 			return NULL;
412cd82e61cSMarcel Holtmann 
413cd82e61cSMarcel Holtmann 		ni = (void *)skb_put(skb, HCI_MON_NEW_INDEX_SIZE);
414cd82e61cSMarcel Holtmann 		ni->type = hdev->dev_type;
415cd82e61cSMarcel Holtmann 		ni->bus = hdev->bus;
416cd82e61cSMarcel Holtmann 		bacpy(&ni->bdaddr, &hdev->bdaddr);
417cd82e61cSMarcel Holtmann 		memcpy(ni->name, hdev->name, 8);
418cd82e61cSMarcel Holtmann 
419dcf4adbfSJoe Perches 		opcode = cpu_to_le16(HCI_MON_NEW_INDEX);
420cd82e61cSMarcel Holtmann 		break;
421cd82e61cSMarcel Holtmann 
422cd82e61cSMarcel Holtmann 	case HCI_DEV_UNREG:
423cd82e61cSMarcel Holtmann 		skb = bt_skb_alloc(0, GFP_ATOMIC);
424cd82e61cSMarcel Holtmann 		if (!skb)
425cd82e61cSMarcel Holtmann 			return NULL;
426cd82e61cSMarcel Holtmann 
427dcf4adbfSJoe Perches 		opcode = cpu_to_le16(HCI_MON_DEL_INDEX);
428cd82e61cSMarcel Holtmann 		break;
429cd82e61cSMarcel Holtmann 
430e131d74aSMarcel Holtmann 	case HCI_DEV_SETUP:
431e131d74aSMarcel Holtmann 		if (hdev->manufacturer == 0xffff)
432e131d74aSMarcel Holtmann 			return NULL;
433e131d74aSMarcel Holtmann 
434e131d74aSMarcel Holtmann 		/* fall through */
435e131d74aSMarcel Holtmann 
4366c566dd5SMarcel Holtmann 	case HCI_DEV_UP:
4376c566dd5SMarcel Holtmann 		skb = bt_skb_alloc(HCI_MON_INDEX_INFO_SIZE, GFP_ATOMIC);
4386c566dd5SMarcel Holtmann 		if (!skb)
4396c566dd5SMarcel Holtmann 			return NULL;
4406c566dd5SMarcel Holtmann 
4416c566dd5SMarcel Holtmann 		ii = (void *)skb_put(skb, HCI_MON_INDEX_INFO_SIZE);
4426c566dd5SMarcel Holtmann 		bacpy(&ii->bdaddr, &hdev->bdaddr);
4436c566dd5SMarcel Holtmann 		ii->manufacturer = cpu_to_le16(hdev->manufacturer);
4446c566dd5SMarcel Holtmann 
4456c566dd5SMarcel Holtmann 		opcode = cpu_to_le16(HCI_MON_INDEX_INFO);
4466c566dd5SMarcel Holtmann 		break;
4476c566dd5SMarcel Holtmann 
44822db3cbcSMarcel Holtmann 	case HCI_DEV_OPEN:
44922db3cbcSMarcel Holtmann 		skb = bt_skb_alloc(0, GFP_ATOMIC);
45022db3cbcSMarcel Holtmann 		if (!skb)
45122db3cbcSMarcel Holtmann 			return NULL;
45222db3cbcSMarcel Holtmann 
45322db3cbcSMarcel Holtmann 		opcode = cpu_to_le16(HCI_MON_OPEN_INDEX);
45422db3cbcSMarcel Holtmann 		break;
45522db3cbcSMarcel Holtmann 
45622db3cbcSMarcel Holtmann 	case HCI_DEV_CLOSE:
45722db3cbcSMarcel Holtmann 		skb = bt_skb_alloc(0, GFP_ATOMIC);
45822db3cbcSMarcel Holtmann 		if (!skb)
45922db3cbcSMarcel Holtmann 			return NULL;
46022db3cbcSMarcel Holtmann 
46122db3cbcSMarcel Holtmann 		opcode = cpu_to_le16(HCI_MON_CLOSE_INDEX);
46222db3cbcSMarcel Holtmann 		break;
46322db3cbcSMarcel Holtmann 
464cd82e61cSMarcel Holtmann 	default:
465cd82e61cSMarcel Holtmann 		return NULL;
466cd82e61cSMarcel Holtmann 	}
467cd82e61cSMarcel Holtmann 
468cd82e61cSMarcel Holtmann 	__net_timestamp(skb);
469cd82e61cSMarcel Holtmann 
470cd82e61cSMarcel Holtmann 	hdr = (void *)skb_push(skb, HCI_MON_HDR_SIZE);
471cd82e61cSMarcel Holtmann 	hdr->opcode = opcode;
472cd82e61cSMarcel Holtmann 	hdr->index = cpu_to_le16(hdev->id);
473cd82e61cSMarcel Holtmann 	hdr->len = cpu_to_le16(skb->len - HCI_MON_HDR_SIZE);
474cd82e61cSMarcel Holtmann 
475cd82e61cSMarcel Holtmann 	return skb;
476cd82e61cSMarcel Holtmann }
477cd82e61cSMarcel Holtmann 
478249fa169SMarcel Holtmann static struct sk_buff *create_monitor_ctrl_open(struct sock *sk)
479249fa169SMarcel Holtmann {
480249fa169SMarcel Holtmann 	struct hci_mon_hdr *hdr;
481249fa169SMarcel Holtmann 	struct sk_buff *skb;
482*d0bef1d2SMarcel Holtmann 	u16 format;
483249fa169SMarcel Holtmann 	u8 ver[3];
484249fa169SMarcel Holtmann 	u32 flags;
485249fa169SMarcel Holtmann 
4860ef2c42fSMarcel Holtmann 	/* No message needed when cookie is not present */
4870ef2c42fSMarcel Holtmann 	if (!hci_pi(sk)->cookie)
4880ef2c42fSMarcel Holtmann 		return NULL;
4890ef2c42fSMarcel Holtmann 
490*d0bef1d2SMarcel Holtmann 	switch (hci_pi(sk)->channel) {
491*d0bef1d2SMarcel Holtmann 	case HCI_CHANNEL_CONTROL:
492*d0bef1d2SMarcel Holtmann 		format = 0x0002;
493*d0bef1d2SMarcel Holtmann 		mgmt_fill_version_info(ver);
494*d0bef1d2SMarcel Holtmann 		break;
495*d0bef1d2SMarcel Holtmann 	default:
496*d0bef1d2SMarcel Holtmann 		/* No message for unsupported format */
497*d0bef1d2SMarcel Holtmann 		return NULL;
498*d0bef1d2SMarcel Holtmann 	}
499*d0bef1d2SMarcel Holtmann 
500249fa169SMarcel Holtmann 	skb = bt_skb_alloc(14 + TASK_COMM_LEN , GFP_ATOMIC);
501249fa169SMarcel Holtmann 	if (!skb)
502249fa169SMarcel Holtmann 		return NULL;
503249fa169SMarcel Holtmann 
504249fa169SMarcel Holtmann 	flags = hci_sock_test_flag(sk, HCI_SOCK_TRUSTED) ? 0x1 : 0x0;
505249fa169SMarcel Holtmann 
506249fa169SMarcel Holtmann 	put_unaligned_le32(hci_pi(sk)->cookie, skb_put(skb, 4));
507249fa169SMarcel Holtmann 	put_unaligned_le16(format, skb_put(skb, 2));
508249fa169SMarcel Holtmann 	memcpy(skb_put(skb, sizeof(ver)), ver, sizeof(ver));
509249fa169SMarcel Holtmann 	put_unaligned_le32(flags, skb_put(skb, 4));
510249fa169SMarcel Holtmann 	*skb_put(skb, 1) = TASK_COMM_LEN;
511249fa169SMarcel Holtmann 	memcpy(skb_put(skb, TASK_COMM_LEN), hci_pi(sk)->comm, TASK_COMM_LEN);
512249fa169SMarcel Holtmann 
513249fa169SMarcel Holtmann 	__net_timestamp(skb);
514249fa169SMarcel Holtmann 
515249fa169SMarcel Holtmann 	hdr = (void *)skb_push(skb, HCI_MON_HDR_SIZE);
516249fa169SMarcel Holtmann 	hdr->opcode = cpu_to_le16(HCI_MON_CTRL_OPEN);
5170ef2c42fSMarcel Holtmann 	if (hci_pi(sk)->hdev)
5180ef2c42fSMarcel Holtmann 		hdr->index = cpu_to_le16(hci_pi(sk)->hdev->id);
5190ef2c42fSMarcel Holtmann 	else
520249fa169SMarcel Holtmann 		hdr->index = cpu_to_le16(HCI_DEV_NONE);
521249fa169SMarcel Holtmann 	hdr->len = cpu_to_le16(skb->len - HCI_MON_HDR_SIZE);
522249fa169SMarcel Holtmann 
523249fa169SMarcel Holtmann 	return skb;
524249fa169SMarcel Holtmann }
525249fa169SMarcel Holtmann 
526249fa169SMarcel Holtmann static struct sk_buff *create_monitor_ctrl_close(struct sock *sk)
527249fa169SMarcel Holtmann {
528249fa169SMarcel Holtmann 	struct hci_mon_hdr *hdr;
529249fa169SMarcel Holtmann 	struct sk_buff *skb;
530249fa169SMarcel Holtmann 
5310ef2c42fSMarcel Holtmann 	/* No message needed when cookie is not present */
5320ef2c42fSMarcel Holtmann 	if (!hci_pi(sk)->cookie)
5330ef2c42fSMarcel Holtmann 		return NULL;
5340ef2c42fSMarcel Holtmann 
535*d0bef1d2SMarcel Holtmann 	switch (hci_pi(sk)->channel) {
536*d0bef1d2SMarcel Holtmann 	case HCI_CHANNEL_CONTROL:
537*d0bef1d2SMarcel Holtmann 		break;
538*d0bef1d2SMarcel Holtmann 	default:
539*d0bef1d2SMarcel Holtmann 		/* No message for unsupported format */
540*d0bef1d2SMarcel Holtmann 		return NULL;
541*d0bef1d2SMarcel Holtmann 	}
542*d0bef1d2SMarcel Holtmann 
543249fa169SMarcel Holtmann 	skb = bt_skb_alloc(4, GFP_ATOMIC);
544249fa169SMarcel Holtmann 	if (!skb)
545249fa169SMarcel Holtmann 		return NULL;
546249fa169SMarcel Holtmann 
547249fa169SMarcel Holtmann 	put_unaligned_le32(hci_pi(sk)->cookie, skb_put(skb, 4));
548249fa169SMarcel Holtmann 
549249fa169SMarcel Holtmann 	__net_timestamp(skb);
550249fa169SMarcel Holtmann 
551249fa169SMarcel Holtmann 	hdr = (void *)skb_push(skb, HCI_MON_HDR_SIZE);
552249fa169SMarcel Holtmann 	hdr->opcode = cpu_to_le16(HCI_MON_CTRL_CLOSE);
5530ef2c42fSMarcel Holtmann 	if (hci_pi(sk)->hdev)
5540ef2c42fSMarcel Holtmann 		hdr->index = cpu_to_le16(hci_pi(sk)->hdev->id);
5550ef2c42fSMarcel Holtmann 	else
556249fa169SMarcel Holtmann 		hdr->index = cpu_to_le16(HCI_DEV_NONE);
557249fa169SMarcel Holtmann 	hdr->len = cpu_to_le16(skb->len - HCI_MON_HDR_SIZE);
558249fa169SMarcel Holtmann 
559249fa169SMarcel Holtmann 	return skb;
560249fa169SMarcel Holtmann }
561249fa169SMarcel Holtmann 
56238ceaa00SMarcel Holtmann static struct sk_buff *create_monitor_ctrl_command(struct sock *sk, u16 index,
56338ceaa00SMarcel Holtmann 						   u16 opcode, u16 len,
56438ceaa00SMarcel Holtmann 						   const void *buf)
56538ceaa00SMarcel Holtmann {
56638ceaa00SMarcel Holtmann 	struct hci_mon_hdr *hdr;
56738ceaa00SMarcel Holtmann 	struct sk_buff *skb;
56838ceaa00SMarcel Holtmann 
56938ceaa00SMarcel Holtmann 	skb = bt_skb_alloc(6 + len, GFP_ATOMIC);
57038ceaa00SMarcel Holtmann 	if (!skb)
57138ceaa00SMarcel Holtmann 		return NULL;
57238ceaa00SMarcel Holtmann 
57338ceaa00SMarcel Holtmann 	put_unaligned_le32(hci_pi(sk)->cookie, skb_put(skb, 4));
57438ceaa00SMarcel Holtmann 	put_unaligned_le16(opcode, skb_put(skb, 2));
57538ceaa00SMarcel Holtmann 
57638ceaa00SMarcel Holtmann 	if (buf)
57738ceaa00SMarcel Holtmann 		memcpy(skb_put(skb, len), buf, len);
57838ceaa00SMarcel Holtmann 
57938ceaa00SMarcel Holtmann 	__net_timestamp(skb);
58038ceaa00SMarcel Holtmann 
58138ceaa00SMarcel Holtmann 	hdr = (void *)skb_push(skb, HCI_MON_HDR_SIZE);
58238ceaa00SMarcel Holtmann 	hdr->opcode = cpu_to_le16(HCI_MON_CTRL_COMMAND);
58338ceaa00SMarcel Holtmann 	hdr->index = cpu_to_le16(index);
58438ceaa00SMarcel Holtmann 	hdr->len = cpu_to_le16(skb->len - HCI_MON_HDR_SIZE);
58538ceaa00SMarcel Holtmann 
58638ceaa00SMarcel Holtmann 	return skb;
58738ceaa00SMarcel Holtmann }
58838ceaa00SMarcel Holtmann 
589787b306cSJohannes Berg static void __printf(2, 3)
590787b306cSJohannes Berg send_monitor_note(struct sock *sk, const char *fmt, ...)
591dd31506dSMarcel Holtmann {
592787b306cSJohannes Berg 	size_t len;
593dd31506dSMarcel Holtmann 	struct hci_mon_hdr *hdr;
594dd31506dSMarcel Holtmann 	struct sk_buff *skb;
595787b306cSJohannes Berg 	va_list args;
596787b306cSJohannes Berg 
597787b306cSJohannes Berg 	va_start(args, fmt);
598787b306cSJohannes Berg 	len = vsnprintf(NULL, 0, fmt, args);
599787b306cSJohannes Berg 	va_end(args);
600dd31506dSMarcel Holtmann 
601dd31506dSMarcel Holtmann 	skb = bt_skb_alloc(len + 1, GFP_ATOMIC);
602dd31506dSMarcel Holtmann 	if (!skb)
603dd31506dSMarcel Holtmann 		return;
604dd31506dSMarcel Holtmann 
605787b306cSJohannes Berg 	va_start(args, fmt);
606787b306cSJohannes Berg 	vsprintf(skb_put(skb, len), fmt, args);
607787b306cSJohannes Berg 	*skb_put(skb, 1) = 0;
608787b306cSJohannes Berg 	va_end(args);
609dd31506dSMarcel Holtmann 
610dd31506dSMarcel Holtmann 	__net_timestamp(skb);
611dd31506dSMarcel Holtmann 
612dd31506dSMarcel Holtmann 	hdr = (void *)skb_push(skb, HCI_MON_HDR_SIZE);
613dd31506dSMarcel Holtmann 	hdr->opcode = cpu_to_le16(HCI_MON_SYSTEM_NOTE);
614dd31506dSMarcel Holtmann 	hdr->index = cpu_to_le16(HCI_DEV_NONE);
615dd31506dSMarcel Holtmann 	hdr->len = cpu_to_le16(skb->len - HCI_MON_HDR_SIZE);
616dd31506dSMarcel Holtmann 
617dd31506dSMarcel Holtmann 	if (sock_queue_rcv_skb(sk, skb))
618dd31506dSMarcel Holtmann 		kfree_skb(skb);
619dd31506dSMarcel Holtmann }
620dd31506dSMarcel Holtmann 
621cd82e61cSMarcel Holtmann static void send_monitor_replay(struct sock *sk)
622cd82e61cSMarcel Holtmann {
623cd82e61cSMarcel Holtmann 	struct hci_dev *hdev;
624cd82e61cSMarcel Holtmann 
625cd82e61cSMarcel Holtmann 	read_lock(&hci_dev_list_lock);
626cd82e61cSMarcel Holtmann 
627cd82e61cSMarcel Holtmann 	list_for_each_entry(hdev, &hci_dev_list, list) {
628cd82e61cSMarcel Holtmann 		struct sk_buff *skb;
629cd82e61cSMarcel Holtmann 
630cd82e61cSMarcel Holtmann 		skb = create_monitor_event(hdev, HCI_DEV_REG);
631cd82e61cSMarcel Holtmann 		if (!skb)
632cd82e61cSMarcel Holtmann 			continue;
633cd82e61cSMarcel Holtmann 
634cd82e61cSMarcel Holtmann 		if (sock_queue_rcv_skb(sk, skb))
635cd82e61cSMarcel Holtmann 			kfree_skb(skb);
63622db3cbcSMarcel Holtmann 
63722db3cbcSMarcel Holtmann 		if (!test_bit(HCI_RUNNING, &hdev->flags))
63822db3cbcSMarcel Holtmann 			continue;
63922db3cbcSMarcel Holtmann 
64022db3cbcSMarcel Holtmann 		skb = create_monitor_event(hdev, HCI_DEV_OPEN);
64122db3cbcSMarcel Holtmann 		if (!skb)
64222db3cbcSMarcel Holtmann 			continue;
64322db3cbcSMarcel Holtmann 
64422db3cbcSMarcel Holtmann 		if (sock_queue_rcv_skb(sk, skb))
64522db3cbcSMarcel Holtmann 			kfree_skb(skb);
6466c566dd5SMarcel Holtmann 
647e131d74aSMarcel Holtmann 		if (test_bit(HCI_UP, &hdev->flags))
6486c566dd5SMarcel Holtmann 			skb = create_monitor_event(hdev, HCI_DEV_UP);
649e131d74aSMarcel Holtmann 		else if (hci_dev_test_flag(hdev, HCI_SETUP))
650e131d74aSMarcel Holtmann 			skb = create_monitor_event(hdev, HCI_DEV_SETUP);
651e131d74aSMarcel Holtmann 		else
652e131d74aSMarcel Holtmann 			skb = NULL;
6536c566dd5SMarcel Holtmann 
654e131d74aSMarcel Holtmann 		if (skb) {
6556c566dd5SMarcel Holtmann 			if (sock_queue_rcv_skb(sk, skb))
6566c566dd5SMarcel Holtmann 				kfree_skb(skb);
657cd82e61cSMarcel Holtmann 		}
658e131d74aSMarcel Holtmann 	}
659cd82e61cSMarcel Holtmann 
660cd82e61cSMarcel Holtmann 	read_unlock(&hci_dev_list_lock);
661cd82e61cSMarcel Holtmann }
662cd82e61cSMarcel Holtmann 
663249fa169SMarcel Holtmann static void send_monitor_control_replay(struct sock *mon_sk)
664249fa169SMarcel Holtmann {
665249fa169SMarcel Holtmann 	struct sock *sk;
666249fa169SMarcel Holtmann 
667249fa169SMarcel Holtmann 	read_lock(&hci_sk_list.lock);
668249fa169SMarcel Holtmann 
669249fa169SMarcel Holtmann 	sk_for_each(sk, &hci_sk_list.head) {
670249fa169SMarcel Holtmann 		struct sk_buff *skb;
671249fa169SMarcel Holtmann 
672249fa169SMarcel Holtmann 		skb = create_monitor_ctrl_open(sk);
673249fa169SMarcel Holtmann 		if (!skb)
674249fa169SMarcel Holtmann 			continue;
675249fa169SMarcel Holtmann 
676249fa169SMarcel Holtmann 		if (sock_queue_rcv_skb(mon_sk, skb))
677249fa169SMarcel Holtmann 			kfree_skb(skb);
678249fa169SMarcel Holtmann 	}
679249fa169SMarcel Holtmann 
680249fa169SMarcel Holtmann 	read_unlock(&hci_sk_list.lock);
681249fa169SMarcel Holtmann }
682249fa169SMarcel Holtmann 
683040030efSMarcel Holtmann /* Generate internal stack event */
684040030efSMarcel Holtmann static void hci_si_event(struct hci_dev *hdev, int type, int dlen, void *data)
685040030efSMarcel Holtmann {
686040030efSMarcel Holtmann 	struct hci_event_hdr *hdr;
687040030efSMarcel Holtmann 	struct hci_ev_stack_internal *ev;
688040030efSMarcel Holtmann 	struct sk_buff *skb;
689040030efSMarcel Holtmann 
690040030efSMarcel Holtmann 	skb = bt_skb_alloc(HCI_EVENT_HDR_SIZE + sizeof(*ev) + dlen, GFP_ATOMIC);
691040030efSMarcel Holtmann 	if (!skb)
692040030efSMarcel Holtmann 		return;
693040030efSMarcel Holtmann 
694040030efSMarcel Holtmann 	hdr = (void *)skb_put(skb, HCI_EVENT_HDR_SIZE);
695040030efSMarcel Holtmann 	hdr->evt  = HCI_EV_STACK_INTERNAL;
696040030efSMarcel Holtmann 	hdr->plen = sizeof(*ev) + dlen;
697040030efSMarcel Holtmann 
698040030efSMarcel Holtmann 	ev  = (void *)skb_put(skb, sizeof(*ev) + dlen);
699040030efSMarcel Holtmann 	ev->type = type;
700040030efSMarcel Holtmann 	memcpy(ev->data, data, dlen);
701040030efSMarcel Holtmann 
702040030efSMarcel Holtmann 	bt_cb(skb)->incoming = 1;
703040030efSMarcel Holtmann 	__net_timestamp(skb);
704040030efSMarcel Holtmann 
705d79f34e3SMarcel Holtmann 	hci_skb_pkt_type(skb) = HCI_EVENT_PKT;
706040030efSMarcel Holtmann 	hci_send_to_sock(hdev, skb);
707040030efSMarcel Holtmann 	kfree_skb(skb);
708040030efSMarcel Holtmann }
709040030efSMarcel Holtmann 
710040030efSMarcel Holtmann void hci_sock_dev_event(struct hci_dev *hdev, int event)
711040030efSMarcel Holtmann {
712040030efSMarcel Holtmann 	BT_DBG("hdev %s event %d", hdev->name, event);
713040030efSMarcel Holtmann 
714cd82e61cSMarcel Holtmann 	if (atomic_read(&monitor_promisc)) {
715cd82e61cSMarcel Holtmann 		struct sk_buff *skb;
716cd82e61cSMarcel Holtmann 
717ed1b28a4SMarcel Holtmann 		/* Send event to monitor */
718cd82e61cSMarcel Holtmann 		skb = create_monitor_event(hdev, event);
719cd82e61cSMarcel Holtmann 		if (skb) {
720c08b1a1dSMarcel Holtmann 			hci_send_to_channel(HCI_CHANNEL_MONITOR, skb,
721c08b1a1dSMarcel Holtmann 					    HCI_SOCK_TRUSTED, NULL);
722cd82e61cSMarcel Holtmann 			kfree_skb(skb);
723cd82e61cSMarcel Holtmann 		}
724cd82e61cSMarcel Holtmann 	}
725cd82e61cSMarcel Holtmann 
726ed1b28a4SMarcel Holtmann 	if (event <= HCI_DEV_DOWN) {
727ed1b28a4SMarcel Holtmann 		struct hci_ev_si_device ev;
728ed1b28a4SMarcel Holtmann 
729040030efSMarcel Holtmann 		/* Send event to sockets */
730040030efSMarcel Holtmann 		ev.event  = event;
731040030efSMarcel Holtmann 		ev.dev_id = hdev->id;
732040030efSMarcel Holtmann 		hci_si_event(NULL, HCI_EV_SI_DEVICE, sizeof(ev), &ev);
733ed1b28a4SMarcel Holtmann 	}
734040030efSMarcel Holtmann 
735040030efSMarcel Holtmann 	if (event == HCI_DEV_UNREG) {
736040030efSMarcel Holtmann 		struct sock *sk;
737040030efSMarcel Holtmann 
738040030efSMarcel Holtmann 		/* Detach sockets from device */
739040030efSMarcel Holtmann 		read_lock(&hci_sk_list.lock);
740b67bfe0dSSasha Levin 		sk_for_each(sk, &hci_sk_list.head) {
741040030efSMarcel Holtmann 			bh_lock_sock_nested(sk);
742040030efSMarcel Holtmann 			if (hci_pi(sk)->hdev == hdev) {
743040030efSMarcel Holtmann 				hci_pi(sk)->hdev = NULL;
744040030efSMarcel Holtmann 				sk->sk_err = EPIPE;
745040030efSMarcel Holtmann 				sk->sk_state = BT_OPEN;
746040030efSMarcel Holtmann 				sk->sk_state_change(sk);
747040030efSMarcel Holtmann 
748040030efSMarcel Holtmann 				hci_dev_put(hdev);
749040030efSMarcel Holtmann 			}
750040030efSMarcel Holtmann 			bh_unlock_sock(sk);
751040030efSMarcel Holtmann 		}
752040030efSMarcel Holtmann 		read_unlock(&hci_sk_list.lock);
753040030efSMarcel Holtmann 	}
754040030efSMarcel Holtmann }
755040030efSMarcel Holtmann 
756801c1e8dSJohan Hedberg static struct hci_mgmt_chan *__hci_mgmt_chan_find(unsigned short channel)
757801c1e8dSJohan Hedberg {
758801c1e8dSJohan Hedberg 	struct hci_mgmt_chan *c;
759801c1e8dSJohan Hedberg 
760801c1e8dSJohan Hedberg 	list_for_each_entry(c, &mgmt_chan_list, list) {
761801c1e8dSJohan Hedberg 		if (c->channel == channel)
762801c1e8dSJohan Hedberg 			return c;
763801c1e8dSJohan Hedberg 	}
764801c1e8dSJohan Hedberg 
765801c1e8dSJohan Hedberg 	return NULL;
766801c1e8dSJohan Hedberg }
767801c1e8dSJohan Hedberg 
768801c1e8dSJohan Hedberg static struct hci_mgmt_chan *hci_mgmt_chan_find(unsigned short channel)
769801c1e8dSJohan Hedberg {
770801c1e8dSJohan Hedberg 	struct hci_mgmt_chan *c;
771801c1e8dSJohan Hedberg 
772801c1e8dSJohan Hedberg 	mutex_lock(&mgmt_chan_list_lock);
773801c1e8dSJohan Hedberg 	c = __hci_mgmt_chan_find(channel);
774801c1e8dSJohan Hedberg 	mutex_unlock(&mgmt_chan_list_lock);
775801c1e8dSJohan Hedberg 
776801c1e8dSJohan Hedberg 	return c;
777801c1e8dSJohan Hedberg }
778801c1e8dSJohan Hedberg 
779801c1e8dSJohan Hedberg int hci_mgmt_chan_register(struct hci_mgmt_chan *c)
780801c1e8dSJohan Hedberg {
781801c1e8dSJohan Hedberg 	if (c->channel < HCI_CHANNEL_CONTROL)
782801c1e8dSJohan Hedberg 		return -EINVAL;
783801c1e8dSJohan Hedberg 
784801c1e8dSJohan Hedberg 	mutex_lock(&mgmt_chan_list_lock);
785801c1e8dSJohan Hedberg 	if (__hci_mgmt_chan_find(c->channel)) {
786801c1e8dSJohan Hedberg 		mutex_unlock(&mgmt_chan_list_lock);
787801c1e8dSJohan Hedberg 		return -EALREADY;
788801c1e8dSJohan Hedberg 	}
789801c1e8dSJohan Hedberg 
790801c1e8dSJohan Hedberg 	list_add_tail(&c->list, &mgmt_chan_list);
791801c1e8dSJohan Hedberg 
792801c1e8dSJohan Hedberg 	mutex_unlock(&mgmt_chan_list_lock);
793801c1e8dSJohan Hedberg 
794801c1e8dSJohan Hedberg 	return 0;
795801c1e8dSJohan Hedberg }
796801c1e8dSJohan Hedberg EXPORT_SYMBOL(hci_mgmt_chan_register);
797801c1e8dSJohan Hedberg 
798801c1e8dSJohan Hedberg void hci_mgmt_chan_unregister(struct hci_mgmt_chan *c)
799801c1e8dSJohan Hedberg {
800801c1e8dSJohan Hedberg 	mutex_lock(&mgmt_chan_list_lock);
801801c1e8dSJohan Hedberg 	list_del(&c->list);
802801c1e8dSJohan Hedberg 	mutex_unlock(&mgmt_chan_list_lock);
803801c1e8dSJohan Hedberg }
804801c1e8dSJohan Hedberg EXPORT_SYMBOL(hci_mgmt_chan_unregister);
805801c1e8dSJohan Hedberg 
8061da177e4SLinus Torvalds static int hci_sock_release(struct socket *sock)
8071da177e4SLinus Torvalds {
8081da177e4SLinus Torvalds 	struct sock *sk = sock->sk;
8097b005bd3SMarcel Holtmann 	struct hci_dev *hdev;
810249fa169SMarcel Holtmann 	struct sk_buff *skb;
8111da177e4SLinus Torvalds 
8121da177e4SLinus Torvalds 	BT_DBG("sock %p sk %p", sock, sk);
8131da177e4SLinus Torvalds 
8141da177e4SLinus Torvalds 	if (!sk)
8151da177e4SLinus Torvalds 		return 0;
8161da177e4SLinus Torvalds 
8177b005bd3SMarcel Holtmann 	hdev = hci_pi(sk)->hdev;
8187b005bd3SMarcel Holtmann 
81970ecce91SMarcel Holtmann 	switch (hci_pi(sk)->channel) {
82070ecce91SMarcel Holtmann 	case HCI_CHANNEL_MONITOR:
821cd82e61cSMarcel Holtmann 		atomic_dec(&monitor_promisc);
82270ecce91SMarcel Holtmann 		break;
82370ecce91SMarcel Holtmann 	case HCI_CHANNEL_CONTROL:
824249fa169SMarcel Holtmann 		/* Send event to monitor */
825249fa169SMarcel Holtmann 		skb = create_monitor_ctrl_close(sk);
826249fa169SMarcel Holtmann 		if (skb) {
827249fa169SMarcel Holtmann 			hci_send_to_channel(HCI_CHANNEL_MONITOR, skb,
828249fa169SMarcel Holtmann 					    HCI_SOCK_TRUSTED, NULL);
829249fa169SMarcel Holtmann 			kfree_skb(skb);
830249fa169SMarcel Holtmann 		}
831249fa169SMarcel Holtmann 
832df1cb87aSMarcel Holtmann 		hci_sock_free_cookie(sk);
83370ecce91SMarcel Holtmann 		break;
83470ecce91SMarcel Holtmann 	}
835cd82e61cSMarcel Holtmann 
8361da177e4SLinus Torvalds 	bt_sock_unlink(&hci_sk_list, sk);
8371da177e4SLinus Torvalds 
8381da177e4SLinus Torvalds 	if (hdev) {
83923500189SMarcel Holtmann 		if (hci_pi(sk)->channel == HCI_CHANNEL_USER) {
8406b3cc1dbSSimon Fels 			/* When releasing an user channel exclusive access,
8416b3cc1dbSSimon Fels 			 * call hci_dev_do_close directly instead of calling
8426b3cc1dbSSimon Fels 			 * hci_dev_close to ensure the exclusive access will
8436b3cc1dbSSimon Fels 			 * be released and the controller brought back down.
8446b3cc1dbSSimon Fels 			 *
8456b3cc1dbSSimon Fels 			 * The checking of HCI_AUTO_OFF is not needed in this
8466b3cc1dbSSimon Fels 			 * case since it will have been cleared already when
8476b3cc1dbSSimon Fels 			 * opening the user channel.
8486b3cc1dbSSimon Fels 			 */
8496b3cc1dbSSimon Fels 			hci_dev_do_close(hdev);
8509380f9eaSLoic Poulain 			hci_dev_clear_flag(hdev, HCI_USER_CHANNEL);
8519380f9eaSLoic Poulain 			mgmt_index_added(hdev);
85223500189SMarcel Holtmann 		}
85323500189SMarcel Holtmann 
8541da177e4SLinus Torvalds 		atomic_dec(&hdev->promisc);
8551da177e4SLinus Torvalds 		hci_dev_put(hdev);
8561da177e4SLinus Torvalds 	}
8571da177e4SLinus Torvalds 
8581da177e4SLinus Torvalds 	sock_orphan(sk);
8591da177e4SLinus Torvalds 
8601da177e4SLinus Torvalds 	skb_queue_purge(&sk->sk_receive_queue);
8611da177e4SLinus Torvalds 	skb_queue_purge(&sk->sk_write_queue);
8621da177e4SLinus Torvalds 
8631da177e4SLinus Torvalds 	sock_put(sk);
8641da177e4SLinus Torvalds 	return 0;
8651da177e4SLinus Torvalds }
8661da177e4SLinus Torvalds 
867b2a66aadSAntti Julku static int hci_sock_blacklist_add(struct hci_dev *hdev, void __user *arg)
868f0358568SJohan Hedberg {
869f0358568SJohan Hedberg 	bdaddr_t bdaddr;
8705e762444SAntti Julku 	int err;
871f0358568SJohan Hedberg 
872f0358568SJohan Hedberg 	if (copy_from_user(&bdaddr, arg, sizeof(bdaddr)))
873f0358568SJohan Hedberg 		return -EFAULT;
874f0358568SJohan Hedberg 
87509fd0de5SGustavo F. Padovan 	hci_dev_lock(hdev);
8765e762444SAntti Julku 
877dcc36c16SJohan Hedberg 	err = hci_bdaddr_list_add(&hdev->blacklist, &bdaddr, BDADDR_BREDR);
8785e762444SAntti Julku 
87909fd0de5SGustavo F. Padovan 	hci_dev_unlock(hdev);
8805e762444SAntti Julku 
8815e762444SAntti Julku 	return err;
882f0358568SJohan Hedberg }
883f0358568SJohan Hedberg 
884b2a66aadSAntti Julku static int hci_sock_blacklist_del(struct hci_dev *hdev, void __user *arg)
885f0358568SJohan Hedberg {
886f0358568SJohan Hedberg 	bdaddr_t bdaddr;
8875e762444SAntti Julku 	int err;
888f0358568SJohan Hedberg 
889f0358568SJohan Hedberg 	if (copy_from_user(&bdaddr, arg, sizeof(bdaddr)))
890f0358568SJohan Hedberg 		return -EFAULT;
891f0358568SJohan Hedberg 
89209fd0de5SGustavo F. Padovan 	hci_dev_lock(hdev);
8935e762444SAntti Julku 
894dcc36c16SJohan Hedberg 	err = hci_bdaddr_list_del(&hdev->blacklist, &bdaddr, BDADDR_BREDR);
8955e762444SAntti Julku 
89609fd0de5SGustavo F. Padovan 	hci_dev_unlock(hdev);
8975e762444SAntti Julku 
8985e762444SAntti Julku 	return err;
899f0358568SJohan Hedberg }
900f0358568SJohan Hedberg 
9011da177e4SLinus Torvalds /* Ioctls that require bound socket */
9026039aa73SGustavo Padovan static int hci_sock_bound_ioctl(struct sock *sk, unsigned int cmd,
9036039aa73SGustavo Padovan 				unsigned long arg)
9041da177e4SLinus Torvalds {
9051da177e4SLinus Torvalds 	struct hci_dev *hdev = hci_pi(sk)->hdev;
9061da177e4SLinus Torvalds 
9071da177e4SLinus Torvalds 	if (!hdev)
9081da177e4SLinus Torvalds 		return -EBADFD;
9091da177e4SLinus Torvalds 
910d7a5a11dSMarcel Holtmann 	if (hci_dev_test_flag(hdev, HCI_USER_CHANNEL))
9110736cfa8SMarcel Holtmann 		return -EBUSY;
9120736cfa8SMarcel Holtmann 
913d7a5a11dSMarcel Holtmann 	if (hci_dev_test_flag(hdev, HCI_UNCONFIGURED))
914fee746b0SMarcel Holtmann 		return -EOPNOTSUPP;
915fee746b0SMarcel Holtmann 
916ca8bee5dSMarcel Holtmann 	if (hdev->dev_type != HCI_PRIMARY)
9175b69bef5SMarcel Holtmann 		return -EOPNOTSUPP;
9185b69bef5SMarcel Holtmann 
9191da177e4SLinus Torvalds 	switch (cmd) {
9201da177e4SLinus Torvalds 	case HCISETRAW:
9211da177e4SLinus Torvalds 		if (!capable(CAP_NET_ADMIN))
922bf5b30b8SZhao Hongjiang 			return -EPERM;
923db596681SMarcel Holtmann 		return -EOPNOTSUPP;
9241da177e4SLinus Torvalds 
9251da177e4SLinus Torvalds 	case HCIGETCONNINFO:
9261da177e4SLinus Torvalds 		return hci_get_conn_info(hdev, (void __user *)arg);
9271da177e4SLinus Torvalds 
92840be492fSMarcel Holtmann 	case HCIGETAUTHINFO:
92940be492fSMarcel Holtmann 		return hci_get_auth_info(hdev, (void __user *)arg);
93040be492fSMarcel Holtmann 
931f0358568SJohan Hedberg 	case HCIBLOCKADDR:
932f0358568SJohan Hedberg 		if (!capable(CAP_NET_ADMIN))
933bf5b30b8SZhao Hongjiang 			return -EPERM;
934b2a66aadSAntti Julku 		return hci_sock_blacklist_add(hdev, (void __user *)arg);
935f0358568SJohan Hedberg 
936f0358568SJohan Hedberg 	case HCIUNBLOCKADDR:
937f0358568SJohan Hedberg 		if (!capable(CAP_NET_ADMIN))
938bf5b30b8SZhao Hongjiang 			return -EPERM;
939b2a66aadSAntti Julku 		return hci_sock_blacklist_del(hdev, (void __user *)arg);
9400736cfa8SMarcel Holtmann 	}
941f0358568SJohan Hedberg 
942324d36edSMarcel Holtmann 	return -ENOIOCTLCMD;
9431da177e4SLinus Torvalds }
9441da177e4SLinus Torvalds 
9458fc9ced3SGustavo Padovan static int hci_sock_ioctl(struct socket *sock, unsigned int cmd,
9468fc9ced3SGustavo Padovan 			  unsigned long arg)
9471da177e4SLinus Torvalds {
9481da177e4SLinus Torvalds 	void __user *argp = (void __user *)arg;
9490736cfa8SMarcel Holtmann 	struct sock *sk = sock->sk;
9501da177e4SLinus Torvalds 	int err;
9511da177e4SLinus Torvalds 
9521da177e4SLinus Torvalds 	BT_DBG("cmd %x arg %lx", cmd, arg);
9531da177e4SLinus Torvalds 
954c1c4f956SMarcel Holtmann 	lock_sock(sk);
955c1c4f956SMarcel Holtmann 
956c1c4f956SMarcel Holtmann 	if (hci_pi(sk)->channel != HCI_CHANNEL_RAW) {
957c1c4f956SMarcel Holtmann 		err = -EBADFD;
958c1c4f956SMarcel Holtmann 		goto done;
959c1c4f956SMarcel Holtmann 	}
960c1c4f956SMarcel Holtmann 
961c1c4f956SMarcel Holtmann 	release_sock(sk);
962c1c4f956SMarcel Holtmann 
9631da177e4SLinus Torvalds 	switch (cmd) {
9641da177e4SLinus Torvalds 	case HCIGETDEVLIST:
9651da177e4SLinus Torvalds 		return hci_get_dev_list(argp);
9661da177e4SLinus Torvalds 
9671da177e4SLinus Torvalds 	case HCIGETDEVINFO:
9681da177e4SLinus Torvalds 		return hci_get_dev_info(argp);
9691da177e4SLinus Torvalds 
9701da177e4SLinus Torvalds 	case HCIGETCONNLIST:
9711da177e4SLinus Torvalds 		return hci_get_conn_list(argp);
9721da177e4SLinus Torvalds 
9731da177e4SLinus Torvalds 	case HCIDEVUP:
9741da177e4SLinus Torvalds 		if (!capable(CAP_NET_ADMIN))
975bf5b30b8SZhao Hongjiang 			return -EPERM;
9761da177e4SLinus Torvalds 		return hci_dev_open(arg);
9771da177e4SLinus Torvalds 
9781da177e4SLinus Torvalds 	case HCIDEVDOWN:
9791da177e4SLinus Torvalds 		if (!capable(CAP_NET_ADMIN))
980bf5b30b8SZhao Hongjiang 			return -EPERM;
9811da177e4SLinus Torvalds 		return hci_dev_close(arg);
9821da177e4SLinus Torvalds 
9831da177e4SLinus Torvalds 	case HCIDEVRESET:
9841da177e4SLinus Torvalds 		if (!capable(CAP_NET_ADMIN))
985bf5b30b8SZhao Hongjiang 			return -EPERM;
9861da177e4SLinus Torvalds 		return hci_dev_reset(arg);
9871da177e4SLinus Torvalds 
9881da177e4SLinus Torvalds 	case HCIDEVRESTAT:
9891da177e4SLinus Torvalds 		if (!capable(CAP_NET_ADMIN))
990bf5b30b8SZhao Hongjiang 			return -EPERM;
9911da177e4SLinus Torvalds 		return hci_dev_reset_stat(arg);
9921da177e4SLinus Torvalds 
9931da177e4SLinus Torvalds 	case HCISETSCAN:
9941da177e4SLinus Torvalds 	case HCISETAUTH:
9951da177e4SLinus Torvalds 	case HCISETENCRYPT:
9961da177e4SLinus Torvalds 	case HCISETPTYPE:
9971da177e4SLinus Torvalds 	case HCISETLINKPOL:
9981da177e4SLinus Torvalds 	case HCISETLINKMODE:
9991da177e4SLinus Torvalds 	case HCISETACLMTU:
10001da177e4SLinus Torvalds 	case HCISETSCOMTU:
10011da177e4SLinus Torvalds 		if (!capable(CAP_NET_ADMIN))
1002bf5b30b8SZhao Hongjiang 			return -EPERM;
10031da177e4SLinus Torvalds 		return hci_dev_cmd(cmd, argp);
10041da177e4SLinus Torvalds 
10051da177e4SLinus Torvalds 	case HCIINQUIRY:
10061da177e4SLinus Torvalds 		return hci_inquiry(argp);
1007c1c4f956SMarcel Holtmann 	}
10081da177e4SLinus Torvalds 
10091da177e4SLinus Torvalds 	lock_sock(sk);
1010c1c4f956SMarcel Holtmann 
10111da177e4SLinus Torvalds 	err = hci_sock_bound_ioctl(sk, cmd, arg);
1012c1c4f956SMarcel Holtmann 
1013c1c4f956SMarcel Holtmann done:
10141da177e4SLinus Torvalds 	release_sock(sk);
10151da177e4SLinus Torvalds 	return err;
10161da177e4SLinus Torvalds }
10171da177e4SLinus Torvalds 
10188fc9ced3SGustavo Padovan static int hci_sock_bind(struct socket *sock, struct sockaddr *addr,
10198fc9ced3SGustavo Padovan 			 int addr_len)
10201da177e4SLinus Torvalds {
10210381101fSJohan Hedberg 	struct sockaddr_hci haddr;
10221da177e4SLinus Torvalds 	struct sock *sk = sock->sk;
10231da177e4SLinus Torvalds 	struct hci_dev *hdev = NULL;
10240381101fSJohan Hedberg 	int len, err = 0;
10251da177e4SLinus Torvalds 
10261da177e4SLinus Torvalds 	BT_DBG("sock %p sk %p", sock, sk);
10271da177e4SLinus Torvalds 
10280381101fSJohan Hedberg 	if (!addr)
10290381101fSJohan Hedberg 		return -EINVAL;
10300381101fSJohan Hedberg 
10310381101fSJohan Hedberg 	memset(&haddr, 0, sizeof(haddr));
10320381101fSJohan Hedberg 	len = min_t(unsigned int, sizeof(haddr), addr_len);
10330381101fSJohan Hedberg 	memcpy(&haddr, addr, len);
10340381101fSJohan Hedberg 
10350381101fSJohan Hedberg 	if (haddr.hci_family != AF_BLUETOOTH)
10360381101fSJohan Hedberg 		return -EINVAL;
10370381101fSJohan Hedberg 
10381da177e4SLinus Torvalds 	lock_sock(sk);
10391da177e4SLinus Torvalds 
10407cc2ade2SMarcel Holtmann 	if (sk->sk_state == BT_BOUND) {
10417cc2ade2SMarcel Holtmann 		err = -EALREADY;
10427cc2ade2SMarcel Holtmann 		goto done;
10437cc2ade2SMarcel Holtmann 	}
10447cc2ade2SMarcel Holtmann 
10457cc2ade2SMarcel Holtmann 	switch (haddr.hci_channel) {
10467cc2ade2SMarcel Holtmann 	case HCI_CHANNEL_RAW:
10477cc2ade2SMarcel Holtmann 		if (hci_pi(sk)->hdev) {
10481da177e4SLinus Torvalds 			err = -EALREADY;
10491da177e4SLinus Torvalds 			goto done;
10501da177e4SLinus Torvalds 		}
10511da177e4SLinus Torvalds 
10520381101fSJohan Hedberg 		if (haddr.hci_dev != HCI_DEV_NONE) {
10530381101fSJohan Hedberg 			hdev = hci_dev_get(haddr.hci_dev);
105470f23020SAndrei Emeltchenko 			if (!hdev) {
10551da177e4SLinus Torvalds 				err = -ENODEV;
10561da177e4SLinus Torvalds 				goto done;
10571da177e4SLinus Torvalds 			}
10581da177e4SLinus Torvalds 
10591da177e4SLinus Torvalds 			atomic_inc(&hdev->promisc);
10601da177e4SLinus Torvalds 		}
10611da177e4SLinus Torvalds 
10625a6d2cf5SMarcel Holtmann 		hci_pi(sk)->channel = haddr.hci_channel;
10631da177e4SLinus Torvalds 		hci_pi(sk)->hdev = hdev;
10647cc2ade2SMarcel Holtmann 		break;
10657cc2ade2SMarcel Holtmann 
106623500189SMarcel Holtmann 	case HCI_CHANNEL_USER:
106723500189SMarcel Holtmann 		if (hci_pi(sk)->hdev) {
106823500189SMarcel Holtmann 			err = -EALREADY;
106923500189SMarcel Holtmann 			goto done;
107023500189SMarcel Holtmann 		}
107123500189SMarcel Holtmann 
107223500189SMarcel Holtmann 		if (haddr.hci_dev == HCI_DEV_NONE) {
107323500189SMarcel Holtmann 			err = -EINVAL;
107423500189SMarcel Holtmann 			goto done;
107523500189SMarcel Holtmann 		}
107623500189SMarcel Holtmann 
107710a8b86fSMarcel Holtmann 		if (!capable(CAP_NET_ADMIN)) {
107823500189SMarcel Holtmann 			err = -EPERM;
107923500189SMarcel Holtmann 			goto done;
108023500189SMarcel Holtmann 		}
108123500189SMarcel Holtmann 
108223500189SMarcel Holtmann 		hdev = hci_dev_get(haddr.hci_dev);
108323500189SMarcel Holtmann 		if (!hdev) {
108423500189SMarcel Holtmann 			err = -ENODEV;
108523500189SMarcel Holtmann 			goto done;
108623500189SMarcel Holtmann 		}
108723500189SMarcel Holtmann 
1088781f899fSMarcel Holtmann 		if (test_bit(HCI_INIT, &hdev->flags) ||
1089d7a5a11dSMarcel Holtmann 		    hci_dev_test_flag(hdev, HCI_SETUP) ||
1090781f899fSMarcel Holtmann 		    hci_dev_test_flag(hdev, HCI_CONFIG) ||
1091781f899fSMarcel Holtmann 		    (!hci_dev_test_flag(hdev, HCI_AUTO_OFF) &&
1092781f899fSMarcel Holtmann 		     test_bit(HCI_UP, &hdev->flags))) {
109323500189SMarcel Holtmann 			err = -EBUSY;
109423500189SMarcel Holtmann 			hci_dev_put(hdev);
109523500189SMarcel Holtmann 			goto done;
109623500189SMarcel Holtmann 		}
109723500189SMarcel Holtmann 
1098238be788SMarcel Holtmann 		if (hci_dev_test_and_set_flag(hdev, HCI_USER_CHANNEL)) {
109923500189SMarcel Holtmann 			err = -EUSERS;
110023500189SMarcel Holtmann 			hci_dev_put(hdev);
110123500189SMarcel Holtmann 			goto done;
110223500189SMarcel Holtmann 		}
110323500189SMarcel Holtmann 
110423500189SMarcel Holtmann 		mgmt_index_removed(hdev);
110523500189SMarcel Holtmann 
110623500189SMarcel Holtmann 		err = hci_dev_open(hdev->id);
110723500189SMarcel Holtmann 		if (err) {
1108781f899fSMarcel Holtmann 			if (err == -EALREADY) {
1109781f899fSMarcel Holtmann 				/* In case the transport is already up and
1110781f899fSMarcel Holtmann 				 * running, clear the error here.
1111781f899fSMarcel Holtmann 				 *
1112781f899fSMarcel Holtmann 				 * This can happen when opening an user
1113781f899fSMarcel Holtmann 				 * channel and HCI_AUTO_OFF grace period
1114781f899fSMarcel Holtmann 				 * is still active.
1115781f899fSMarcel Holtmann 				 */
1116781f899fSMarcel Holtmann 				err = 0;
1117781f899fSMarcel Holtmann 			} else {
1118a358dc11SMarcel Holtmann 				hci_dev_clear_flag(hdev, HCI_USER_CHANNEL);
1119c6521401SMarcel Holtmann 				mgmt_index_added(hdev);
112023500189SMarcel Holtmann 				hci_dev_put(hdev);
112123500189SMarcel Holtmann 				goto done;
112223500189SMarcel Holtmann 			}
1123781f899fSMarcel Holtmann 		}
112423500189SMarcel Holtmann 
11255a6d2cf5SMarcel Holtmann 		hci_pi(sk)->channel = haddr.hci_channel;
112623500189SMarcel Holtmann 		hci_pi(sk)->hdev = hdev;
11275a6d2cf5SMarcel Holtmann 
11285a6d2cf5SMarcel Holtmann 		atomic_inc(&hdev->promisc);
112923500189SMarcel Holtmann 		break;
113023500189SMarcel Holtmann 
1131cd82e61cSMarcel Holtmann 	case HCI_CHANNEL_MONITOR:
1132cd82e61cSMarcel Holtmann 		if (haddr.hci_dev != HCI_DEV_NONE) {
1133cd82e61cSMarcel Holtmann 			err = -EINVAL;
1134cd82e61cSMarcel Holtmann 			goto done;
1135cd82e61cSMarcel Holtmann 		}
1136cd82e61cSMarcel Holtmann 
1137cd82e61cSMarcel Holtmann 		if (!capable(CAP_NET_RAW)) {
1138cd82e61cSMarcel Holtmann 			err = -EPERM;
1139cd82e61cSMarcel Holtmann 			goto done;
1140cd82e61cSMarcel Holtmann 		}
1141cd82e61cSMarcel Holtmann 
11425a6d2cf5SMarcel Holtmann 		hci_pi(sk)->channel = haddr.hci_channel;
11435a6d2cf5SMarcel Holtmann 
114450ebc055SMarcel Holtmann 		/* The monitor interface is restricted to CAP_NET_RAW
114550ebc055SMarcel Holtmann 		 * capabilities and with that implicitly trusted.
114650ebc055SMarcel Holtmann 		 */
114750ebc055SMarcel Holtmann 		hci_sock_set_flag(sk, HCI_SOCK_TRUSTED);
114850ebc055SMarcel Holtmann 
1149787b306cSJohannes Berg 		send_monitor_note(sk, "Linux version %s (%s)",
1150787b306cSJohannes Berg 				  init_utsname()->release,
1151787b306cSJohannes Berg 				  init_utsname()->machine);
11529e8305b3SMarcel Holtmann 		send_monitor_note(sk, "Bluetooth subsystem version %u.%u",
11539e8305b3SMarcel Holtmann 				  BT_SUBSYS_VERSION, BT_SUBSYS_REVISION);
1154cd82e61cSMarcel Holtmann 		send_monitor_replay(sk);
1155249fa169SMarcel Holtmann 		send_monitor_control_replay(sk);
1156cd82e61cSMarcel Holtmann 
1157cd82e61cSMarcel Holtmann 		atomic_inc(&monitor_promisc);
1158cd82e61cSMarcel Holtmann 		break;
1159cd82e61cSMarcel Holtmann 
1160ac714949SMarcel Holtmann 	case HCI_CHANNEL_LOGGING:
1161ac714949SMarcel Holtmann 		if (haddr.hci_dev != HCI_DEV_NONE) {
1162ac714949SMarcel Holtmann 			err = -EINVAL;
1163ac714949SMarcel Holtmann 			goto done;
1164ac714949SMarcel Holtmann 		}
1165ac714949SMarcel Holtmann 
1166ac714949SMarcel Holtmann 		if (!capable(CAP_NET_ADMIN)) {
1167ac714949SMarcel Holtmann 			err = -EPERM;
1168ac714949SMarcel Holtmann 			goto done;
1169ac714949SMarcel Holtmann 		}
11705a6d2cf5SMarcel Holtmann 
11715a6d2cf5SMarcel Holtmann 		hci_pi(sk)->channel = haddr.hci_channel;
1172ac714949SMarcel Holtmann 		break;
1173ac714949SMarcel Holtmann 
11747cc2ade2SMarcel Holtmann 	default:
1175801c1e8dSJohan Hedberg 		if (!hci_mgmt_chan_find(haddr.hci_channel)) {
11767cc2ade2SMarcel Holtmann 			err = -EINVAL;
11777cc2ade2SMarcel Holtmann 			goto done;
11787cc2ade2SMarcel Holtmann 		}
11797cc2ade2SMarcel Holtmann 
1180801c1e8dSJohan Hedberg 		if (haddr.hci_dev != HCI_DEV_NONE) {
1181801c1e8dSJohan Hedberg 			err = -EINVAL;
1182801c1e8dSJohan Hedberg 			goto done;
1183801c1e8dSJohan Hedberg 		}
1184801c1e8dSJohan Hedberg 
11851195fbb8SMarcel Holtmann 		/* Users with CAP_NET_ADMIN capabilities are allowed
11861195fbb8SMarcel Holtmann 		 * access to all management commands and events. For
11871195fbb8SMarcel Holtmann 		 * untrusted users the interface is restricted and
11881195fbb8SMarcel Holtmann 		 * also only untrusted events are sent.
118950ebc055SMarcel Holtmann 		 */
11901195fbb8SMarcel Holtmann 		if (capable(CAP_NET_ADMIN))
119150ebc055SMarcel Holtmann 			hci_sock_set_flag(sk, HCI_SOCK_TRUSTED);
119250ebc055SMarcel Holtmann 
11935a6d2cf5SMarcel Holtmann 		hci_pi(sk)->channel = haddr.hci_channel;
11945a6d2cf5SMarcel Holtmann 
1195f9207338SMarcel Holtmann 		/* At the moment the index and unconfigured index events
1196f9207338SMarcel Holtmann 		 * are enabled unconditionally. Setting them on each
1197f9207338SMarcel Holtmann 		 * socket when binding keeps this functionality. They
1198f9207338SMarcel Holtmann 		 * however might be cleared later and then sending of these
1199f9207338SMarcel Holtmann 		 * events will be disabled, but that is then intentional.
1200f6b7712eSMarcel Holtmann 		 *
1201f6b7712eSMarcel Holtmann 		 * This also enables generic events that are safe to be
1202f6b7712eSMarcel Holtmann 		 * received by untrusted users. Example for such events
1203f6b7712eSMarcel Holtmann 		 * are changes to settings, class of device, name etc.
1204f9207338SMarcel Holtmann 		 */
12055a6d2cf5SMarcel Holtmann 		if (hci_pi(sk)->channel == HCI_CHANNEL_CONTROL) {
1206249fa169SMarcel Holtmann 			struct sk_buff *skb;
120770ecce91SMarcel Holtmann 
1208df1cb87aSMarcel Holtmann 			hci_sock_gen_cookie(sk);
120970ecce91SMarcel Holtmann 
1210249fa169SMarcel Holtmann 			/* Send event to monitor */
1211249fa169SMarcel Holtmann 			skb = create_monitor_ctrl_open(sk);
1212249fa169SMarcel Holtmann 			if (skb) {
1213249fa169SMarcel Holtmann 				hci_send_to_channel(HCI_CHANNEL_MONITOR, skb,
1214249fa169SMarcel Holtmann 						    HCI_SOCK_TRUSTED, NULL);
1215249fa169SMarcel Holtmann 				kfree_skb(skb);
1216249fa169SMarcel Holtmann 			}
1217249fa169SMarcel Holtmann 
1218f9207338SMarcel Holtmann 			hci_sock_set_flag(sk, HCI_MGMT_INDEX_EVENTS);
1219f9207338SMarcel Holtmann 			hci_sock_set_flag(sk, HCI_MGMT_UNCONF_INDEX_EVENTS);
12205504c3a3SMarcel Holtmann 			hci_sock_set_flag(sk, HCI_MGMT_OPTION_EVENTS);
12215504c3a3SMarcel Holtmann 			hci_sock_set_flag(sk, HCI_MGMT_SETTING_EVENTS);
12225504c3a3SMarcel Holtmann 			hci_sock_set_flag(sk, HCI_MGMT_DEV_CLASS_EVENTS);
12235504c3a3SMarcel Holtmann 			hci_sock_set_flag(sk, HCI_MGMT_LOCAL_NAME_EVENTS);
1224f9207338SMarcel Holtmann 		}
1225801c1e8dSJohan Hedberg 		break;
1226801c1e8dSJohan Hedberg 	}
1227801c1e8dSJohan Hedberg 
12281da177e4SLinus Torvalds 	sk->sk_state = BT_BOUND;
12291da177e4SLinus Torvalds 
12301da177e4SLinus Torvalds done:
12311da177e4SLinus Torvalds 	release_sock(sk);
12321da177e4SLinus Torvalds 	return err;
12331da177e4SLinus Torvalds }
12341da177e4SLinus Torvalds 
12358fc9ced3SGustavo Padovan static int hci_sock_getname(struct socket *sock, struct sockaddr *addr,
12368fc9ced3SGustavo Padovan 			    int *addr_len, int peer)
12371da177e4SLinus Torvalds {
12381da177e4SLinus Torvalds 	struct sockaddr_hci *haddr = (struct sockaddr_hci *)addr;
12391da177e4SLinus Torvalds 	struct sock *sk = sock->sk;
12409d4b68b2SMarcel Holtmann 	struct hci_dev *hdev;
12419d4b68b2SMarcel Holtmann 	int err = 0;
12421da177e4SLinus Torvalds 
12431da177e4SLinus Torvalds 	BT_DBG("sock %p sk %p", sock, sk);
12441da177e4SLinus Torvalds 
124506f43cbcSMarcel Holtmann 	if (peer)
124606f43cbcSMarcel Holtmann 		return -EOPNOTSUPP;
124706f43cbcSMarcel Holtmann 
12481da177e4SLinus Torvalds 	lock_sock(sk);
12491da177e4SLinus Torvalds 
12509d4b68b2SMarcel Holtmann 	hdev = hci_pi(sk)->hdev;
12519d4b68b2SMarcel Holtmann 	if (!hdev) {
12529d4b68b2SMarcel Holtmann 		err = -EBADFD;
12539d4b68b2SMarcel Holtmann 		goto done;
12549d4b68b2SMarcel Holtmann 	}
12559d4b68b2SMarcel Holtmann 
12561da177e4SLinus Torvalds 	*addr_len = sizeof(*haddr);
12571da177e4SLinus Torvalds 	haddr->hci_family = AF_BLUETOOTH;
12587b005bd3SMarcel Holtmann 	haddr->hci_dev    = hdev->id;
12599d4b68b2SMarcel Holtmann 	haddr->hci_channel= hci_pi(sk)->channel;
12601da177e4SLinus Torvalds 
12619d4b68b2SMarcel Holtmann done:
12621da177e4SLinus Torvalds 	release_sock(sk);
12639d4b68b2SMarcel Holtmann 	return err;
12641da177e4SLinus Torvalds }
12651da177e4SLinus Torvalds 
12666039aa73SGustavo Padovan static void hci_sock_cmsg(struct sock *sk, struct msghdr *msg,
12676039aa73SGustavo Padovan 			  struct sk_buff *skb)
12681da177e4SLinus Torvalds {
12691da177e4SLinus Torvalds 	__u32 mask = hci_pi(sk)->cmsg_mask;
12701da177e4SLinus Torvalds 
12710d48d939SMarcel Holtmann 	if (mask & HCI_CMSG_DIR) {
12720d48d939SMarcel Holtmann 		int incoming = bt_cb(skb)->incoming;
12738fc9ced3SGustavo Padovan 		put_cmsg(msg, SOL_HCI, HCI_CMSG_DIR, sizeof(incoming),
12748fc9ced3SGustavo Padovan 			 &incoming);
12750d48d939SMarcel Holtmann 	}
12761da177e4SLinus Torvalds 
1277a61bbcf2SPatrick McHardy 	if (mask & HCI_CMSG_TSTAMP) {
1278f6e623a6SJohann Felix Soden #ifdef CONFIG_COMPAT
1279f6e623a6SJohann Felix Soden 		struct compat_timeval ctv;
1280f6e623a6SJohann Felix Soden #endif
1281a61bbcf2SPatrick McHardy 		struct timeval tv;
1282767c5eb5SMarcel Holtmann 		void *data;
1283767c5eb5SMarcel Holtmann 		int len;
1284a61bbcf2SPatrick McHardy 
1285a61bbcf2SPatrick McHardy 		skb_get_timestamp(skb, &tv);
1286767c5eb5SMarcel Holtmann 
12871da97f83SDavid S. Miller 		data = &tv;
12881da97f83SDavid S. Miller 		len = sizeof(tv);
12891da97f83SDavid S. Miller #ifdef CONFIG_COMPAT
1290da88cea1SH. J. Lu 		if (!COMPAT_USE_64BIT_TIME &&
1291da88cea1SH. J. Lu 		    (msg->msg_flags & MSG_CMSG_COMPAT)) {
1292767c5eb5SMarcel Holtmann 			ctv.tv_sec = tv.tv_sec;
1293767c5eb5SMarcel Holtmann 			ctv.tv_usec = tv.tv_usec;
1294767c5eb5SMarcel Holtmann 			data = &ctv;
1295767c5eb5SMarcel Holtmann 			len = sizeof(ctv);
1296767c5eb5SMarcel Holtmann 		}
12971da97f83SDavid S. Miller #endif
1298767c5eb5SMarcel Holtmann 
1299767c5eb5SMarcel Holtmann 		put_cmsg(msg, SOL_HCI, HCI_CMSG_TSTAMP, len, data);
1300a61bbcf2SPatrick McHardy 	}
13011da177e4SLinus Torvalds }
13021da177e4SLinus Torvalds 
13038528d3f7SMarcel Holtmann static int hci_sock_recvmsg(struct socket *sock, struct msghdr *msg,
13048528d3f7SMarcel Holtmann 			    size_t len, int flags)
13051da177e4SLinus Torvalds {
13061da177e4SLinus Torvalds 	int noblock = flags & MSG_DONTWAIT;
13071da177e4SLinus Torvalds 	struct sock *sk = sock->sk;
13081da177e4SLinus Torvalds 	struct sk_buff *skb;
13091da177e4SLinus Torvalds 	int copied, err;
131083871f8cSDenis Kenzior 	unsigned int skblen;
13111da177e4SLinus Torvalds 
13121da177e4SLinus Torvalds 	BT_DBG("sock %p, sk %p", sock, sk);
13131da177e4SLinus Torvalds 
1314d94a6104SMarcel Holtmann 	if (flags & MSG_OOB)
13151da177e4SLinus Torvalds 		return -EOPNOTSUPP;
13161da177e4SLinus Torvalds 
1317ac714949SMarcel Holtmann 	if (hci_pi(sk)->channel == HCI_CHANNEL_LOGGING)
1318ac714949SMarcel Holtmann 		return -EOPNOTSUPP;
1319ac714949SMarcel Holtmann 
13201da177e4SLinus Torvalds 	if (sk->sk_state == BT_CLOSED)
13211da177e4SLinus Torvalds 		return 0;
13221da177e4SLinus Torvalds 
132370f23020SAndrei Emeltchenko 	skb = skb_recv_datagram(sk, flags, noblock, &err);
132470f23020SAndrei Emeltchenko 	if (!skb)
13251da177e4SLinus Torvalds 		return err;
13261da177e4SLinus Torvalds 
132783871f8cSDenis Kenzior 	skblen = skb->len;
13281da177e4SLinus Torvalds 	copied = skb->len;
13291da177e4SLinus Torvalds 	if (len < copied) {
13301da177e4SLinus Torvalds 		msg->msg_flags |= MSG_TRUNC;
13311da177e4SLinus Torvalds 		copied = len;
13321da177e4SLinus Torvalds 	}
13331da177e4SLinus Torvalds 
1334badff6d0SArnaldo Carvalho de Melo 	skb_reset_transport_header(skb);
133551f3d02bSDavid S. Miller 	err = skb_copy_datagram_msg(skb, 0, msg, copied);
13361da177e4SLinus Torvalds 
13373a208627SMarcel Holtmann 	switch (hci_pi(sk)->channel) {
13383a208627SMarcel Holtmann 	case HCI_CHANNEL_RAW:
13391da177e4SLinus Torvalds 		hci_sock_cmsg(sk, msg, skb);
13403a208627SMarcel Holtmann 		break;
134123500189SMarcel Holtmann 	case HCI_CHANNEL_USER:
1342cd82e61cSMarcel Holtmann 	case HCI_CHANNEL_MONITOR:
1343cd82e61cSMarcel Holtmann 		sock_recv_timestamp(msg, sk, skb);
1344cd82e61cSMarcel Holtmann 		break;
1345801c1e8dSJohan Hedberg 	default:
1346801c1e8dSJohan Hedberg 		if (hci_mgmt_chan_find(hci_pi(sk)->channel))
1347801c1e8dSJohan Hedberg 			sock_recv_timestamp(msg, sk, skb);
1348801c1e8dSJohan Hedberg 		break;
13493a208627SMarcel Holtmann 	}
13501da177e4SLinus Torvalds 
13511da177e4SLinus Torvalds 	skb_free_datagram(sk, skb);
13521da177e4SLinus Torvalds 
13534f34228bSLuiz Augusto von Dentz 	if (flags & MSG_TRUNC)
135483871f8cSDenis Kenzior 		copied = skblen;
135583871f8cSDenis Kenzior 
13561da177e4SLinus Torvalds 	return err ? : copied;
13571da177e4SLinus Torvalds }
13581da177e4SLinus Torvalds 
1359fa4335d7SJohan Hedberg static int hci_mgmt_cmd(struct hci_mgmt_chan *chan, struct sock *sk,
1360fa4335d7SJohan Hedberg 			struct msghdr *msg, size_t msglen)
1361fa4335d7SJohan Hedberg {
1362fa4335d7SJohan Hedberg 	void *buf;
1363fa4335d7SJohan Hedberg 	u8 *cp;
1364fa4335d7SJohan Hedberg 	struct mgmt_hdr *hdr;
1365fa4335d7SJohan Hedberg 	u16 opcode, index, len;
1366fa4335d7SJohan Hedberg 	struct hci_dev *hdev = NULL;
1367fa4335d7SJohan Hedberg 	const struct hci_mgmt_handler *handler;
1368fa4335d7SJohan Hedberg 	bool var_len, no_hdev;
1369fa4335d7SJohan Hedberg 	int err;
1370fa4335d7SJohan Hedberg 
1371fa4335d7SJohan Hedberg 	BT_DBG("got %zu bytes", msglen);
1372fa4335d7SJohan Hedberg 
1373fa4335d7SJohan Hedberg 	if (msglen < sizeof(*hdr))
1374fa4335d7SJohan Hedberg 		return -EINVAL;
1375fa4335d7SJohan Hedberg 
1376fa4335d7SJohan Hedberg 	buf = kmalloc(msglen, GFP_KERNEL);
1377fa4335d7SJohan Hedberg 	if (!buf)
1378fa4335d7SJohan Hedberg 		return -ENOMEM;
1379fa4335d7SJohan Hedberg 
1380fa4335d7SJohan Hedberg 	if (memcpy_from_msg(buf, msg, msglen)) {
1381fa4335d7SJohan Hedberg 		err = -EFAULT;
1382fa4335d7SJohan Hedberg 		goto done;
1383fa4335d7SJohan Hedberg 	}
1384fa4335d7SJohan Hedberg 
1385fa4335d7SJohan Hedberg 	hdr = buf;
1386fa4335d7SJohan Hedberg 	opcode = __le16_to_cpu(hdr->opcode);
1387fa4335d7SJohan Hedberg 	index = __le16_to_cpu(hdr->index);
1388fa4335d7SJohan Hedberg 	len = __le16_to_cpu(hdr->len);
1389fa4335d7SJohan Hedberg 
1390fa4335d7SJohan Hedberg 	if (len != msglen - sizeof(*hdr)) {
1391fa4335d7SJohan Hedberg 		err = -EINVAL;
1392fa4335d7SJohan Hedberg 		goto done;
1393fa4335d7SJohan Hedberg 	}
1394fa4335d7SJohan Hedberg 
139538ceaa00SMarcel Holtmann 	if (chan->channel == HCI_CHANNEL_CONTROL) {
139638ceaa00SMarcel Holtmann 		struct sk_buff *skb;
139738ceaa00SMarcel Holtmann 
139838ceaa00SMarcel Holtmann 		/* Send event to monitor */
139938ceaa00SMarcel Holtmann 		skb = create_monitor_ctrl_command(sk, index, opcode, len,
140038ceaa00SMarcel Holtmann 						  buf + sizeof(*hdr));
140138ceaa00SMarcel Holtmann 		if (skb) {
140238ceaa00SMarcel Holtmann 			hci_send_to_channel(HCI_CHANNEL_MONITOR, skb,
140338ceaa00SMarcel Holtmann 					    HCI_SOCK_TRUSTED, NULL);
140438ceaa00SMarcel Holtmann 			kfree_skb(skb);
140538ceaa00SMarcel Holtmann 		}
140638ceaa00SMarcel Holtmann 	}
140738ceaa00SMarcel Holtmann 
1408fa4335d7SJohan Hedberg 	if (opcode >= chan->handler_count ||
1409fa4335d7SJohan Hedberg 	    chan->handlers[opcode].func == NULL) {
1410fa4335d7SJohan Hedberg 		BT_DBG("Unknown op %u", opcode);
1411fa4335d7SJohan Hedberg 		err = mgmt_cmd_status(sk, index, opcode,
1412fa4335d7SJohan Hedberg 				      MGMT_STATUS_UNKNOWN_COMMAND);
1413fa4335d7SJohan Hedberg 		goto done;
1414fa4335d7SJohan Hedberg 	}
1415fa4335d7SJohan Hedberg 
1416fa4335d7SJohan Hedberg 	handler = &chan->handlers[opcode];
1417fa4335d7SJohan Hedberg 
1418fa4335d7SJohan Hedberg 	if (!hci_sock_test_flag(sk, HCI_SOCK_TRUSTED) &&
1419fa4335d7SJohan Hedberg 	    !(handler->flags & HCI_MGMT_UNTRUSTED)) {
1420fa4335d7SJohan Hedberg 		err = mgmt_cmd_status(sk, index, opcode,
1421fa4335d7SJohan Hedberg 				      MGMT_STATUS_PERMISSION_DENIED);
1422fa4335d7SJohan Hedberg 		goto done;
1423fa4335d7SJohan Hedberg 	}
1424fa4335d7SJohan Hedberg 
1425fa4335d7SJohan Hedberg 	if (index != MGMT_INDEX_NONE) {
1426fa4335d7SJohan Hedberg 		hdev = hci_dev_get(index);
1427fa4335d7SJohan Hedberg 		if (!hdev) {
1428fa4335d7SJohan Hedberg 			err = mgmt_cmd_status(sk, index, opcode,
1429fa4335d7SJohan Hedberg 					      MGMT_STATUS_INVALID_INDEX);
1430fa4335d7SJohan Hedberg 			goto done;
1431fa4335d7SJohan Hedberg 		}
1432fa4335d7SJohan Hedberg 
1433fa4335d7SJohan Hedberg 		if (hci_dev_test_flag(hdev, HCI_SETUP) ||
1434fa4335d7SJohan Hedberg 		    hci_dev_test_flag(hdev, HCI_CONFIG) ||
1435fa4335d7SJohan Hedberg 		    hci_dev_test_flag(hdev, HCI_USER_CHANNEL)) {
1436fa4335d7SJohan Hedberg 			err = mgmt_cmd_status(sk, index, opcode,
1437fa4335d7SJohan Hedberg 					      MGMT_STATUS_INVALID_INDEX);
1438fa4335d7SJohan Hedberg 			goto done;
1439fa4335d7SJohan Hedberg 		}
1440fa4335d7SJohan Hedberg 
1441fa4335d7SJohan Hedberg 		if (hci_dev_test_flag(hdev, HCI_UNCONFIGURED) &&
1442fa4335d7SJohan Hedberg 		    !(handler->flags & HCI_MGMT_UNCONFIGURED)) {
1443fa4335d7SJohan Hedberg 			err = mgmt_cmd_status(sk, index, opcode,
1444fa4335d7SJohan Hedberg 					      MGMT_STATUS_INVALID_INDEX);
1445fa4335d7SJohan Hedberg 			goto done;
1446fa4335d7SJohan Hedberg 		}
1447fa4335d7SJohan Hedberg 	}
1448fa4335d7SJohan Hedberg 
1449fa4335d7SJohan Hedberg 	no_hdev = (handler->flags & HCI_MGMT_NO_HDEV);
1450fa4335d7SJohan Hedberg 	if (no_hdev != !hdev) {
1451fa4335d7SJohan Hedberg 		err = mgmt_cmd_status(sk, index, opcode,
1452fa4335d7SJohan Hedberg 				      MGMT_STATUS_INVALID_INDEX);
1453fa4335d7SJohan Hedberg 		goto done;
1454fa4335d7SJohan Hedberg 	}
1455fa4335d7SJohan Hedberg 
1456fa4335d7SJohan Hedberg 	var_len = (handler->flags & HCI_MGMT_VAR_LEN);
1457fa4335d7SJohan Hedberg 	if ((var_len && len < handler->data_len) ||
1458fa4335d7SJohan Hedberg 	    (!var_len && len != handler->data_len)) {
1459fa4335d7SJohan Hedberg 		err = mgmt_cmd_status(sk, index, opcode,
1460fa4335d7SJohan Hedberg 				      MGMT_STATUS_INVALID_PARAMS);
1461fa4335d7SJohan Hedberg 		goto done;
1462fa4335d7SJohan Hedberg 	}
1463fa4335d7SJohan Hedberg 
1464fa4335d7SJohan Hedberg 	if (hdev && chan->hdev_init)
1465fa4335d7SJohan Hedberg 		chan->hdev_init(sk, hdev);
1466fa4335d7SJohan Hedberg 
1467fa4335d7SJohan Hedberg 	cp = buf + sizeof(*hdr);
1468fa4335d7SJohan Hedberg 
1469fa4335d7SJohan Hedberg 	err = handler->func(sk, hdev, cp, len);
1470fa4335d7SJohan Hedberg 	if (err < 0)
1471fa4335d7SJohan Hedberg 		goto done;
1472fa4335d7SJohan Hedberg 
1473fa4335d7SJohan Hedberg 	err = msglen;
1474fa4335d7SJohan Hedberg 
1475fa4335d7SJohan Hedberg done:
1476fa4335d7SJohan Hedberg 	if (hdev)
1477fa4335d7SJohan Hedberg 		hci_dev_put(hdev);
1478fa4335d7SJohan Hedberg 
1479fa4335d7SJohan Hedberg 	kfree(buf);
1480fa4335d7SJohan Hedberg 	return err;
1481fa4335d7SJohan Hedberg }
1482fa4335d7SJohan Hedberg 
1483ac714949SMarcel Holtmann static int hci_logging_frame(struct sock *sk, struct msghdr *msg, int len)
1484ac714949SMarcel Holtmann {
1485ac714949SMarcel Holtmann 	struct hci_mon_hdr *hdr;
1486ac714949SMarcel Holtmann 	struct sk_buff *skb;
1487ac714949SMarcel Holtmann 	struct hci_dev *hdev;
1488ac714949SMarcel Holtmann 	u16 index;
1489ac714949SMarcel Holtmann 	int err;
1490ac714949SMarcel Holtmann 
1491ac714949SMarcel Holtmann 	/* The logging frame consists at minimum of the standard header,
1492ac714949SMarcel Holtmann 	 * the priority byte, the ident length byte and at least one string
1493ac714949SMarcel Holtmann 	 * terminator NUL byte. Anything shorter are invalid packets.
1494ac714949SMarcel Holtmann 	 */
1495ac714949SMarcel Holtmann 	if (len < sizeof(*hdr) + 3)
1496ac714949SMarcel Holtmann 		return -EINVAL;
1497ac714949SMarcel Holtmann 
1498ac714949SMarcel Holtmann 	skb = bt_skb_send_alloc(sk, len, msg->msg_flags & MSG_DONTWAIT, &err);
1499ac714949SMarcel Holtmann 	if (!skb)
1500ac714949SMarcel Holtmann 		return err;
1501ac714949SMarcel Holtmann 
1502ac714949SMarcel Holtmann 	if (memcpy_from_msg(skb_put(skb, len), msg, len)) {
1503ac714949SMarcel Holtmann 		err = -EFAULT;
1504ac714949SMarcel Holtmann 		goto drop;
1505ac714949SMarcel Holtmann 	}
1506ac714949SMarcel Holtmann 
1507ac714949SMarcel Holtmann 	hdr = (void *)skb->data;
1508ac714949SMarcel Holtmann 
1509ac714949SMarcel Holtmann 	if (__le16_to_cpu(hdr->len) != len - sizeof(*hdr)) {
1510ac714949SMarcel Holtmann 		err = -EINVAL;
1511ac714949SMarcel Holtmann 		goto drop;
1512ac714949SMarcel Holtmann 	}
1513ac714949SMarcel Holtmann 
1514ac714949SMarcel Holtmann 	if (__le16_to_cpu(hdr->opcode) == 0x0000) {
1515ac714949SMarcel Holtmann 		__u8 priority = skb->data[sizeof(*hdr)];
1516ac714949SMarcel Holtmann 		__u8 ident_len = skb->data[sizeof(*hdr) + 1];
1517ac714949SMarcel Holtmann 
1518ac714949SMarcel Holtmann 		/* Only the priorities 0-7 are valid and with that any other
1519ac714949SMarcel Holtmann 		 * value results in an invalid packet.
1520ac714949SMarcel Holtmann 		 *
1521ac714949SMarcel Holtmann 		 * The priority byte is followed by an ident length byte and
1522ac714949SMarcel Holtmann 		 * the NUL terminated ident string. Check that the ident
1523ac714949SMarcel Holtmann 		 * length is not overflowing the packet and also that the
1524ac714949SMarcel Holtmann 		 * ident string itself is NUL terminated. In case the ident
1525ac714949SMarcel Holtmann 		 * length is zero, the length value actually doubles as NUL
1526ac714949SMarcel Holtmann 		 * terminator identifier.
1527ac714949SMarcel Holtmann 		 *
1528ac714949SMarcel Holtmann 		 * The message follows the ident string (if present) and
1529ac714949SMarcel Holtmann 		 * must be NUL terminated. Otherwise it is not a valid packet.
1530ac714949SMarcel Holtmann 		 */
1531ac714949SMarcel Holtmann 		if (priority > 7 || skb->data[len - 1] != 0x00 ||
1532ac714949SMarcel Holtmann 		    ident_len > len - sizeof(*hdr) - 3 ||
1533ac714949SMarcel Holtmann 		    skb->data[sizeof(*hdr) + ident_len + 1] != 0x00) {
1534ac714949SMarcel Holtmann 			err = -EINVAL;
1535ac714949SMarcel Holtmann 			goto drop;
1536ac714949SMarcel Holtmann 		}
1537ac714949SMarcel Holtmann 	} else {
1538ac714949SMarcel Holtmann 		err = -EINVAL;
1539ac714949SMarcel Holtmann 		goto drop;
1540ac714949SMarcel Holtmann 	}
1541ac714949SMarcel Holtmann 
1542ac714949SMarcel Holtmann 	index = __le16_to_cpu(hdr->index);
1543ac714949SMarcel Holtmann 
1544ac714949SMarcel Holtmann 	if (index != MGMT_INDEX_NONE) {
1545ac714949SMarcel Holtmann 		hdev = hci_dev_get(index);
1546ac714949SMarcel Holtmann 		if (!hdev) {
1547ac714949SMarcel Holtmann 			err = -ENODEV;
1548ac714949SMarcel Holtmann 			goto drop;
1549ac714949SMarcel Holtmann 		}
1550ac714949SMarcel Holtmann 	} else {
1551ac714949SMarcel Holtmann 		hdev = NULL;
1552ac714949SMarcel Holtmann 	}
1553ac714949SMarcel Holtmann 
1554ac714949SMarcel Holtmann 	hdr->opcode = cpu_to_le16(HCI_MON_USER_LOGGING);
1555ac714949SMarcel Holtmann 
1556ac714949SMarcel Holtmann 	hci_send_to_channel(HCI_CHANNEL_MONITOR, skb, HCI_SOCK_TRUSTED, NULL);
1557ac714949SMarcel Holtmann 	err = len;
1558ac714949SMarcel Holtmann 
1559ac714949SMarcel Holtmann 	if (hdev)
1560ac714949SMarcel Holtmann 		hci_dev_put(hdev);
1561ac714949SMarcel Holtmann 
1562ac714949SMarcel Holtmann drop:
1563ac714949SMarcel Holtmann 	kfree_skb(skb);
1564ac714949SMarcel Holtmann 	return err;
1565ac714949SMarcel Holtmann }
1566ac714949SMarcel Holtmann 
15671b784140SYing Xue static int hci_sock_sendmsg(struct socket *sock, struct msghdr *msg,
15681b784140SYing Xue 			    size_t len)
15691da177e4SLinus Torvalds {
15701da177e4SLinus Torvalds 	struct sock *sk = sock->sk;
1571801c1e8dSJohan Hedberg 	struct hci_mgmt_chan *chan;
15721da177e4SLinus Torvalds 	struct hci_dev *hdev;
15731da177e4SLinus Torvalds 	struct sk_buff *skb;
15741da177e4SLinus Torvalds 	int err;
15751da177e4SLinus Torvalds 
15761da177e4SLinus Torvalds 	BT_DBG("sock %p sk %p", sock, sk);
15771da177e4SLinus Torvalds 
15781da177e4SLinus Torvalds 	if (msg->msg_flags & MSG_OOB)
15791da177e4SLinus Torvalds 		return -EOPNOTSUPP;
15801da177e4SLinus Torvalds 
15811da177e4SLinus Torvalds 	if (msg->msg_flags & ~(MSG_DONTWAIT|MSG_NOSIGNAL|MSG_ERRQUEUE))
15821da177e4SLinus Torvalds 		return -EINVAL;
15831da177e4SLinus Torvalds 
15841da177e4SLinus Torvalds 	if (len < 4 || len > HCI_MAX_FRAME_SIZE)
15851da177e4SLinus Torvalds 		return -EINVAL;
15861da177e4SLinus Torvalds 
15871da177e4SLinus Torvalds 	lock_sock(sk);
15881da177e4SLinus Torvalds 
15890381101fSJohan Hedberg 	switch (hci_pi(sk)->channel) {
15900381101fSJohan Hedberg 	case HCI_CHANNEL_RAW:
159123500189SMarcel Holtmann 	case HCI_CHANNEL_USER:
15920381101fSJohan Hedberg 		break;
1593cd82e61cSMarcel Holtmann 	case HCI_CHANNEL_MONITOR:
1594cd82e61cSMarcel Holtmann 		err = -EOPNOTSUPP;
1595cd82e61cSMarcel Holtmann 		goto done;
1596ac714949SMarcel Holtmann 	case HCI_CHANNEL_LOGGING:
1597ac714949SMarcel Holtmann 		err = hci_logging_frame(sk, msg, len);
1598ac714949SMarcel Holtmann 		goto done;
15990381101fSJohan Hedberg 	default:
1600801c1e8dSJohan Hedberg 		mutex_lock(&mgmt_chan_list_lock);
1601801c1e8dSJohan Hedberg 		chan = __hci_mgmt_chan_find(hci_pi(sk)->channel);
1602801c1e8dSJohan Hedberg 		if (chan)
1603fa4335d7SJohan Hedberg 			err = hci_mgmt_cmd(chan, sk, msg, len);
1604801c1e8dSJohan Hedberg 		else
16050381101fSJohan Hedberg 			err = -EINVAL;
1606801c1e8dSJohan Hedberg 
1607801c1e8dSJohan Hedberg 		mutex_unlock(&mgmt_chan_list_lock);
16080381101fSJohan Hedberg 		goto done;
16090381101fSJohan Hedberg 	}
16100381101fSJohan Hedberg 
161170f23020SAndrei Emeltchenko 	hdev = hci_pi(sk)->hdev;
161270f23020SAndrei Emeltchenko 	if (!hdev) {
16131da177e4SLinus Torvalds 		err = -EBADFD;
16141da177e4SLinus Torvalds 		goto done;
16151da177e4SLinus Torvalds 	}
16161da177e4SLinus Torvalds 
16177e21addcSMarcel Holtmann 	if (!test_bit(HCI_UP, &hdev->flags)) {
16187e21addcSMarcel Holtmann 		err = -ENETDOWN;
16197e21addcSMarcel Holtmann 		goto done;
16207e21addcSMarcel Holtmann 	}
16217e21addcSMarcel Holtmann 
162270f23020SAndrei Emeltchenko 	skb = bt_skb_send_alloc(sk, len, msg->msg_flags & MSG_DONTWAIT, &err);
162370f23020SAndrei Emeltchenko 	if (!skb)
16241da177e4SLinus Torvalds 		goto done;
16251da177e4SLinus Torvalds 
16266ce8e9ceSAl Viro 	if (memcpy_from_msg(skb_put(skb, len), msg, len)) {
16271da177e4SLinus Torvalds 		err = -EFAULT;
16281da177e4SLinus Torvalds 		goto drop;
16291da177e4SLinus Torvalds 	}
16301da177e4SLinus Torvalds 
16318528d3f7SMarcel Holtmann 	hci_skb_pkt_type(skb) = skb->data[0];
16321da177e4SLinus Torvalds 	skb_pull(skb, 1);
16331da177e4SLinus Torvalds 
16341bc5ad16SMarcel Holtmann 	if (hci_pi(sk)->channel == HCI_CHANNEL_USER) {
16351bc5ad16SMarcel Holtmann 		/* No permission check is needed for user channel
16361bc5ad16SMarcel Holtmann 		 * since that gets enforced when binding the socket.
16371bc5ad16SMarcel Holtmann 		 *
16381bc5ad16SMarcel Holtmann 		 * However check that the packet type is valid.
16391bc5ad16SMarcel Holtmann 		 */
1640d79f34e3SMarcel Holtmann 		if (hci_skb_pkt_type(skb) != HCI_COMMAND_PKT &&
1641d79f34e3SMarcel Holtmann 		    hci_skb_pkt_type(skb) != HCI_ACLDATA_PKT &&
1642d79f34e3SMarcel Holtmann 		    hci_skb_pkt_type(skb) != HCI_SCODATA_PKT) {
16431bc5ad16SMarcel Holtmann 			err = -EINVAL;
16441bc5ad16SMarcel Holtmann 			goto drop;
16451bc5ad16SMarcel Holtmann 		}
16461bc5ad16SMarcel Holtmann 
16471bc5ad16SMarcel Holtmann 		skb_queue_tail(&hdev->raw_q, skb);
16481bc5ad16SMarcel Holtmann 		queue_work(hdev->workqueue, &hdev->tx_work);
1649d79f34e3SMarcel Holtmann 	} else if (hci_skb_pkt_type(skb) == HCI_COMMAND_PKT) {
165083985319SHarvey Harrison 		u16 opcode = get_unaligned_le16(skb->data);
16511da177e4SLinus Torvalds 		u16 ogf = hci_opcode_ogf(opcode);
16521da177e4SLinus Torvalds 		u16 ocf = hci_opcode_ocf(opcode);
16531da177e4SLinus Torvalds 
16541da177e4SLinus Torvalds 		if (((ogf > HCI_SFLT_MAX_OGF) ||
16553bb3c755SGustavo Padovan 		     !hci_test_bit(ocf & HCI_FLT_OCF_BITS,
16563bb3c755SGustavo Padovan 				   &hci_sec_filter.ocf_mask[ogf])) &&
16571da177e4SLinus Torvalds 		    !capable(CAP_NET_RAW)) {
16581da177e4SLinus Torvalds 			err = -EPERM;
16591da177e4SLinus Torvalds 			goto drop;
16601da177e4SLinus Torvalds 		}
16611da177e4SLinus Torvalds 
16621982162bSMarcel Holtmann 		/* Since the opcode has already been extracted here, store
16631982162bSMarcel Holtmann 		 * a copy of the value for later use by the drivers.
16641982162bSMarcel Holtmann 		 */
16651982162bSMarcel Holtmann 		hci_skb_opcode(skb) = opcode;
16661982162bSMarcel Holtmann 
1667fee746b0SMarcel Holtmann 		if (ogf == 0x3f) {
16681da177e4SLinus Torvalds 			skb_queue_tail(&hdev->raw_q, skb);
16693eff45eaSGustavo F. Padovan 			queue_work(hdev->workqueue, &hdev->tx_work);
16701da177e4SLinus Torvalds 		} else {
167149c922bbSStephen Hemminger 			/* Stand-alone HCI commands must be flagged as
167211714b3dSJohan Hedberg 			 * single-command requests.
167311714b3dSJohan Hedberg 			 */
167444d27137SJohan Hedberg 			bt_cb(skb)->hci.req_flags |= HCI_REQ_START;
167511714b3dSJohan Hedberg 
16761da177e4SLinus Torvalds 			skb_queue_tail(&hdev->cmd_q, skb);
1677c347b765SGustavo F. Padovan 			queue_work(hdev->workqueue, &hdev->cmd_work);
16781da177e4SLinus Torvalds 		}
16791da177e4SLinus Torvalds 	} else {
16801da177e4SLinus Torvalds 		if (!capable(CAP_NET_RAW)) {
16811da177e4SLinus Torvalds 			err = -EPERM;
16821da177e4SLinus Torvalds 			goto drop;
16831da177e4SLinus Torvalds 		}
16841da177e4SLinus Torvalds 
1685d79f34e3SMarcel Holtmann 		if (hci_skb_pkt_type(skb) != HCI_ACLDATA_PKT &&
1686d79f34e3SMarcel Holtmann 		    hci_skb_pkt_type(skb) != HCI_SCODATA_PKT) {
1687bb77543eSMarcel Holtmann 			err = -EINVAL;
1688bb77543eSMarcel Holtmann 			goto drop;
1689bb77543eSMarcel Holtmann 		}
1690bb77543eSMarcel Holtmann 
16911da177e4SLinus Torvalds 		skb_queue_tail(&hdev->raw_q, skb);
16923eff45eaSGustavo F. Padovan 		queue_work(hdev->workqueue, &hdev->tx_work);
16931da177e4SLinus Torvalds 	}
16941da177e4SLinus Torvalds 
16951da177e4SLinus Torvalds 	err = len;
16961da177e4SLinus Torvalds 
16971da177e4SLinus Torvalds done:
16981da177e4SLinus Torvalds 	release_sock(sk);
16991da177e4SLinus Torvalds 	return err;
17001da177e4SLinus Torvalds 
17011da177e4SLinus Torvalds drop:
17021da177e4SLinus Torvalds 	kfree_skb(skb);
17031da177e4SLinus Torvalds 	goto done;
17041da177e4SLinus Torvalds }
17051da177e4SLinus Torvalds 
17068fc9ced3SGustavo Padovan static int hci_sock_setsockopt(struct socket *sock, int level, int optname,
17078fc9ced3SGustavo Padovan 			       char __user *optval, unsigned int len)
17081da177e4SLinus Torvalds {
17091da177e4SLinus Torvalds 	struct hci_ufilter uf = { .opcode = 0 };
17101da177e4SLinus Torvalds 	struct sock *sk = sock->sk;
17111da177e4SLinus Torvalds 	int err = 0, opt = 0;
17121da177e4SLinus Torvalds 
17131da177e4SLinus Torvalds 	BT_DBG("sk %p, opt %d", sk, optname);
17141da177e4SLinus Torvalds 
171547b0f573SMarcel Holtmann 	if (level != SOL_HCI)
171647b0f573SMarcel Holtmann 		return -ENOPROTOOPT;
171747b0f573SMarcel Holtmann 
17181da177e4SLinus Torvalds 	lock_sock(sk);
17191da177e4SLinus Torvalds 
17202f39cdb7SMarcel Holtmann 	if (hci_pi(sk)->channel != HCI_CHANNEL_RAW) {
1721c2371e80SMarcel Holtmann 		err = -EBADFD;
17222f39cdb7SMarcel Holtmann 		goto done;
17232f39cdb7SMarcel Holtmann 	}
17242f39cdb7SMarcel Holtmann 
17251da177e4SLinus Torvalds 	switch (optname) {
17261da177e4SLinus Torvalds 	case HCI_DATA_DIR:
17271da177e4SLinus Torvalds 		if (get_user(opt, (int __user *)optval)) {
17281da177e4SLinus Torvalds 			err = -EFAULT;
17291da177e4SLinus Torvalds 			break;
17301da177e4SLinus Torvalds 		}
17311da177e4SLinus Torvalds 
17321da177e4SLinus Torvalds 		if (opt)
17331da177e4SLinus Torvalds 			hci_pi(sk)->cmsg_mask |= HCI_CMSG_DIR;
17341da177e4SLinus Torvalds 		else
17351da177e4SLinus Torvalds 			hci_pi(sk)->cmsg_mask &= ~HCI_CMSG_DIR;
17361da177e4SLinus Torvalds 		break;
17371da177e4SLinus Torvalds 
17381da177e4SLinus Torvalds 	case HCI_TIME_STAMP:
17391da177e4SLinus Torvalds 		if (get_user(opt, (int __user *)optval)) {
17401da177e4SLinus Torvalds 			err = -EFAULT;
17411da177e4SLinus Torvalds 			break;
17421da177e4SLinus Torvalds 		}
17431da177e4SLinus Torvalds 
17441da177e4SLinus Torvalds 		if (opt)
17451da177e4SLinus Torvalds 			hci_pi(sk)->cmsg_mask |= HCI_CMSG_TSTAMP;
17461da177e4SLinus Torvalds 		else
17471da177e4SLinus Torvalds 			hci_pi(sk)->cmsg_mask &= ~HCI_CMSG_TSTAMP;
17481da177e4SLinus Torvalds 		break;
17491da177e4SLinus Torvalds 
17501da177e4SLinus Torvalds 	case HCI_FILTER:
17510878b666SMarcel Holtmann 		{
17520878b666SMarcel Holtmann 			struct hci_filter *f = &hci_pi(sk)->filter;
17530878b666SMarcel Holtmann 
17540878b666SMarcel Holtmann 			uf.type_mask = f->type_mask;
17550878b666SMarcel Holtmann 			uf.opcode    = f->opcode;
17560878b666SMarcel Holtmann 			uf.event_mask[0] = *((u32 *) f->event_mask + 0);
17570878b666SMarcel Holtmann 			uf.event_mask[1] = *((u32 *) f->event_mask + 1);
17580878b666SMarcel Holtmann 		}
17590878b666SMarcel Holtmann 
17601da177e4SLinus Torvalds 		len = min_t(unsigned int, len, sizeof(uf));
17611da177e4SLinus Torvalds 		if (copy_from_user(&uf, optval, len)) {
17621da177e4SLinus Torvalds 			err = -EFAULT;
17631da177e4SLinus Torvalds 			break;
17641da177e4SLinus Torvalds 		}
17651da177e4SLinus Torvalds 
17661da177e4SLinus Torvalds 		if (!capable(CAP_NET_RAW)) {
17671da177e4SLinus Torvalds 			uf.type_mask &= hci_sec_filter.type_mask;
17681da177e4SLinus Torvalds 			uf.event_mask[0] &= *((u32 *) hci_sec_filter.event_mask + 0);
17691da177e4SLinus Torvalds 			uf.event_mask[1] &= *((u32 *) hci_sec_filter.event_mask + 1);
17701da177e4SLinus Torvalds 		}
17711da177e4SLinus Torvalds 
17721da177e4SLinus Torvalds 		{
17731da177e4SLinus Torvalds 			struct hci_filter *f = &hci_pi(sk)->filter;
17741da177e4SLinus Torvalds 
17751da177e4SLinus Torvalds 			f->type_mask = uf.type_mask;
17761da177e4SLinus Torvalds 			f->opcode    = uf.opcode;
17771da177e4SLinus Torvalds 			*((u32 *) f->event_mask + 0) = uf.event_mask[0];
17781da177e4SLinus Torvalds 			*((u32 *) f->event_mask + 1) = uf.event_mask[1];
17791da177e4SLinus Torvalds 		}
17801da177e4SLinus Torvalds 		break;
17811da177e4SLinus Torvalds 
17821da177e4SLinus Torvalds 	default:
17831da177e4SLinus Torvalds 		err = -ENOPROTOOPT;
17841da177e4SLinus Torvalds 		break;
17851da177e4SLinus Torvalds 	}
17861da177e4SLinus Torvalds 
17872f39cdb7SMarcel Holtmann done:
17881da177e4SLinus Torvalds 	release_sock(sk);
17891da177e4SLinus Torvalds 	return err;
17901da177e4SLinus Torvalds }
17911da177e4SLinus Torvalds 
17928fc9ced3SGustavo Padovan static int hci_sock_getsockopt(struct socket *sock, int level, int optname,
17938fc9ced3SGustavo Padovan 			       char __user *optval, int __user *optlen)
17941da177e4SLinus Torvalds {
17951da177e4SLinus Torvalds 	struct hci_ufilter uf;
17961da177e4SLinus Torvalds 	struct sock *sk = sock->sk;
1797cedc5469SMarcel Holtmann 	int len, opt, err = 0;
1798cedc5469SMarcel Holtmann 
1799cedc5469SMarcel Holtmann 	BT_DBG("sk %p, opt %d", sk, optname);
18001da177e4SLinus Torvalds 
180147b0f573SMarcel Holtmann 	if (level != SOL_HCI)
180247b0f573SMarcel Holtmann 		return -ENOPROTOOPT;
180347b0f573SMarcel Holtmann 
18041da177e4SLinus Torvalds 	if (get_user(len, optlen))
18051da177e4SLinus Torvalds 		return -EFAULT;
18061da177e4SLinus Torvalds 
1807cedc5469SMarcel Holtmann 	lock_sock(sk);
1808cedc5469SMarcel Holtmann 
1809cedc5469SMarcel Holtmann 	if (hci_pi(sk)->channel != HCI_CHANNEL_RAW) {
1810c2371e80SMarcel Holtmann 		err = -EBADFD;
1811cedc5469SMarcel Holtmann 		goto done;
1812cedc5469SMarcel Holtmann 	}
1813cedc5469SMarcel Holtmann 
18141da177e4SLinus Torvalds 	switch (optname) {
18151da177e4SLinus Torvalds 	case HCI_DATA_DIR:
18161da177e4SLinus Torvalds 		if (hci_pi(sk)->cmsg_mask & HCI_CMSG_DIR)
18171da177e4SLinus Torvalds 			opt = 1;
18181da177e4SLinus Torvalds 		else
18191da177e4SLinus Torvalds 			opt = 0;
18201da177e4SLinus Torvalds 
18211da177e4SLinus Torvalds 		if (put_user(opt, optval))
1822cedc5469SMarcel Holtmann 			err = -EFAULT;
18231da177e4SLinus Torvalds 		break;
18241da177e4SLinus Torvalds 
18251da177e4SLinus Torvalds 	case HCI_TIME_STAMP:
18261da177e4SLinus Torvalds 		if (hci_pi(sk)->cmsg_mask & HCI_CMSG_TSTAMP)
18271da177e4SLinus Torvalds 			opt = 1;
18281da177e4SLinus Torvalds 		else
18291da177e4SLinus Torvalds 			opt = 0;
18301da177e4SLinus Torvalds 
18311da177e4SLinus Torvalds 		if (put_user(opt, optval))
1832cedc5469SMarcel Holtmann 			err = -EFAULT;
18331da177e4SLinus Torvalds 		break;
18341da177e4SLinus Torvalds 
18351da177e4SLinus Torvalds 	case HCI_FILTER:
18361da177e4SLinus Torvalds 		{
18371da177e4SLinus Torvalds 			struct hci_filter *f = &hci_pi(sk)->filter;
18381da177e4SLinus Torvalds 
1839e15ca9a0SMathias Krause 			memset(&uf, 0, sizeof(uf));
18401da177e4SLinus Torvalds 			uf.type_mask = f->type_mask;
18411da177e4SLinus Torvalds 			uf.opcode    = f->opcode;
18421da177e4SLinus Torvalds 			uf.event_mask[0] = *((u32 *) f->event_mask + 0);
18431da177e4SLinus Torvalds 			uf.event_mask[1] = *((u32 *) f->event_mask + 1);
18441da177e4SLinus Torvalds 		}
18451da177e4SLinus Torvalds 
18461da177e4SLinus Torvalds 		len = min_t(unsigned int, len, sizeof(uf));
18471da177e4SLinus Torvalds 		if (copy_to_user(optval, &uf, len))
1848cedc5469SMarcel Holtmann 			err = -EFAULT;
18491da177e4SLinus Torvalds 		break;
18501da177e4SLinus Torvalds 
18511da177e4SLinus Torvalds 	default:
1852cedc5469SMarcel Holtmann 		err = -ENOPROTOOPT;
18531da177e4SLinus Torvalds 		break;
18541da177e4SLinus Torvalds 	}
18551da177e4SLinus Torvalds 
1856cedc5469SMarcel Holtmann done:
1857cedc5469SMarcel Holtmann 	release_sock(sk);
1858cedc5469SMarcel Holtmann 	return err;
18591da177e4SLinus Torvalds }
18601da177e4SLinus Torvalds 
186190ddc4f0SEric Dumazet static const struct proto_ops hci_sock_ops = {
18621da177e4SLinus Torvalds 	.family		= PF_BLUETOOTH,
18631da177e4SLinus Torvalds 	.owner		= THIS_MODULE,
18641da177e4SLinus Torvalds 	.release	= hci_sock_release,
18651da177e4SLinus Torvalds 	.bind		= hci_sock_bind,
18661da177e4SLinus Torvalds 	.getname	= hci_sock_getname,
18671da177e4SLinus Torvalds 	.sendmsg	= hci_sock_sendmsg,
18681da177e4SLinus Torvalds 	.recvmsg	= hci_sock_recvmsg,
18691da177e4SLinus Torvalds 	.ioctl		= hci_sock_ioctl,
18701da177e4SLinus Torvalds 	.poll		= datagram_poll,
18711da177e4SLinus Torvalds 	.listen		= sock_no_listen,
18721da177e4SLinus Torvalds 	.shutdown	= sock_no_shutdown,
18731da177e4SLinus Torvalds 	.setsockopt	= hci_sock_setsockopt,
18741da177e4SLinus Torvalds 	.getsockopt	= hci_sock_getsockopt,
18751da177e4SLinus Torvalds 	.connect	= sock_no_connect,
18761da177e4SLinus Torvalds 	.socketpair	= sock_no_socketpair,
18771da177e4SLinus Torvalds 	.accept		= sock_no_accept,
18781da177e4SLinus Torvalds 	.mmap		= sock_no_mmap
18791da177e4SLinus Torvalds };
18801da177e4SLinus Torvalds 
18811da177e4SLinus Torvalds static struct proto hci_sk_proto = {
18821da177e4SLinus Torvalds 	.name		= "HCI",
18831da177e4SLinus Torvalds 	.owner		= THIS_MODULE,
18841da177e4SLinus Torvalds 	.obj_size	= sizeof(struct hci_pinfo)
18851da177e4SLinus Torvalds };
18861da177e4SLinus Torvalds 
18873f378b68SEric Paris static int hci_sock_create(struct net *net, struct socket *sock, int protocol,
18883f378b68SEric Paris 			   int kern)
18891da177e4SLinus Torvalds {
18901da177e4SLinus Torvalds 	struct sock *sk;
18911da177e4SLinus Torvalds 
18921da177e4SLinus Torvalds 	BT_DBG("sock %p", sock);
18931da177e4SLinus Torvalds 
18941da177e4SLinus Torvalds 	if (sock->type != SOCK_RAW)
18951da177e4SLinus Torvalds 		return -ESOCKTNOSUPPORT;
18961da177e4SLinus Torvalds 
18971da177e4SLinus Torvalds 	sock->ops = &hci_sock_ops;
18981da177e4SLinus Torvalds 
189911aa9c28SEric W. Biederman 	sk = sk_alloc(net, PF_BLUETOOTH, GFP_ATOMIC, &hci_sk_proto, kern);
19001da177e4SLinus Torvalds 	if (!sk)
19011da177e4SLinus Torvalds 		return -ENOMEM;
19021da177e4SLinus Torvalds 
19031da177e4SLinus Torvalds 	sock_init_data(sock, sk);
19041da177e4SLinus Torvalds 
19051da177e4SLinus Torvalds 	sock_reset_flag(sk, SOCK_ZAPPED);
19061da177e4SLinus Torvalds 
19071da177e4SLinus Torvalds 	sk->sk_protocol = protocol;
19081da177e4SLinus Torvalds 
19091da177e4SLinus Torvalds 	sock->state = SS_UNCONNECTED;
19101da177e4SLinus Torvalds 	sk->sk_state = BT_OPEN;
19111da177e4SLinus Torvalds 
19121da177e4SLinus Torvalds 	bt_sock_link(&hci_sk_list, sk);
19131da177e4SLinus Torvalds 	return 0;
19141da177e4SLinus Torvalds }
19151da177e4SLinus Torvalds 
1916ec1b4cf7SStephen Hemminger static const struct net_proto_family hci_sock_family_ops = {
19171da177e4SLinus Torvalds 	.family	= PF_BLUETOOTH,
19181da177e4SLinus Torvalds 	.owner	= THIS_MODULE,
19191da177e4SLinus Torvalds 	.create	= hci_sock_create,
19201da177e4SLinus Torvalds };
19211da177e4SLinus Torvalds 
19221da177e4SLinus Torvalds int __init hci_sock_init(void)
19231da177e4SLinus Torvalds {
19241da177e4SLinus Torvalds 	int err;
19251da177e4SLinus Torvalds 
1926b0a8e282SMarcel Holtmann 	BUILD_BUG_ON(sizeof(struct sockaddr_hci) > sizeof(struct sockaddr));
1927b0a8e282SMarcel Holtmann 
19281da177e4SLinus Torvalds 	err = proto_register(&hci_sk_proto, 0);
19291da177e4SLinus Torvalds 	if (err < 0)
19301da177e4SLinus Torvalds 		return err;
19311da177e4SLinus Torvalds 
19321da177e4SLinus Torvalds 	err = bt_sock_register(BTPROTO_HCI, &hci_sock_family_ops);
1933f7c86637SMasatake YAMATO 	if (err < 0) {
1934f7c86637SMasatake YAMATO 		BT_ERR("HCI socket registration failed");
19351da177e4SLinus Torvalds 		goto error;
1936f7c86637SMasatake YAMATO 	}
1937f7c86637SMasatake YAMATO 
1938b0316615SAl Viro 	err = bt_procfs_init(&init_net, "hci", &hci_sk_list, NULL);
1939f7c86637SMasatake YAMATO 	if (err < 0) {
1940f7c86637SMasatake YAMATO 		BT_ERR("Failed to create HCI proc file");
1941f7c86637SMasatake YAMATO 		bt_sock_unregister(BTPROTO_HCI);
1942f7c86637SMasatake YAMATO 		goto error;
1943f7c86637SMasatake YAMATO 	}
19441da177e4SLinus Torvalds 
19451da177e4SLinus Torvalds 	BT_INFO("HCI socket layer initialized");
19461da177e4SLinus Torvalds 
19471da177e4SLinus Torvalds 	return 0;
19481da177e4SLinus Torvalds 
19491da177e4SLinus Torvalds error:
19501da177e4SLinus Torvalds 	proto_unregister(&hci_sk_proto);
19511da177e4SLinus Torvalds 	return err;
19521da177e4SLinus Torvalds }
19531da177e4SLinus Torvalds 
1954b7440a14SAnand Gadiyar void hci_sock_cleanup(void)
19551da177e4SLinus Torvalds {
1956f7c86637SMasatake YAMATO 	bt_procfs_cleanup(&init_net, "hci");
19575e9d7f86SDavid Herrmann 	bt_sock_unregister(BTPROTO_HCI);
19581da177e4SLinus Torvalds 	proto_unregister(&hci_sk_proto);
19591da177e4SLinus Torvalds }
1960