xref: /openbmc/linux/net/bluetooth/hci_sock.c (revision 5a6d2cf5f18b5afbae0b1b450070bbba50f1e3e0)
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;
482249fa169SMarcel Holtmann 	u16 format = 0x0002;
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 
490249fa169SMarcel Holtmann 	skb = bt_skb_alloc(14 + TASK_COMM_LEN , GFP_ATOMIC);
491249fa169SMarcel Holtmann 	if (!skb)
492249fa169SMarcel Holtmann 		return NULL;
493249fa169SMarcel Holtmann 
494249fa169SMarcel Holtmann 	mgmt_fill_version_info(ver);
495249fa169SMarcel Holtmann 	flags = hci_sock_test_flag(sk, HCI_SOCK_TRUSTED) ? 0x1 : 0x0;
496249fa169SMarcel Holtmann 
497249fa169SMarcel Holtmann 	put_unaligned_le32(hci_pi(sk)->cookie, skb_put(skb, 4));
498249fa169SMarcel Holtmann 	put_unaligned_le16(format, skb_put(skb, 2));
499249fa169SMarcel Holtmann 	memcpy(skb_put(skb, sizeof(ver)), ver, sizeof(ver));
500249fa169SMarcel Holtmann 	put_unaligned_le32(flags, skb_put(skb, 4));
501249fa169SMarcel Holtmann 	*skb_put(skb, 1) = TASK_COMM_LEN;
502249fa169SMarcel Holtmann 	memcpy(skb_put(skb, TASK_COMM_LEN), hci_pi(sk)->comm, TASK_COMM_LEN);
503249fa169SMarcel Holtmann 
504249fa169SMarcel Holtmann 	__net_timestamp(skb);
505249fa169SMarcel Holtmann 
506249fa169SMarcel Holtmann 	hdr = (void *)skb_push(skb, HCI_MON_HDR_SIZE);
507249fa169SMarcel Holtmann 	hdr->opcode = cpu_to_le16(HCI_MON_CTRL_OPEN);
5080ef2c42fSMarcel Holtmann 	if (hci_pi(sk)->hdev)
5090ef2c42fSMarcel Holtmann 		hdr->index = cpu_to_le16(hci_pi(sk)->hdev->id);
5100ef2c42fSMarcel Holtmann 	else
511249fa169SMarcel Holtmann 		hdr->index = cpu_to_le16(HCI_DEV_NONE);
512249fa169SMarcel Holtmann 	hdr->len = cpu_to_le16(skb->len - HCI_MON_HDR_SIZE);
513249fa169SMarcel Holtmann 
514249fa169SMarcel Holtmann 	return skb;
515249fa169SMarcel Holtmann }
516249fa169SMarcel Holtmann 
517249fa169SMarcel Holtmann static struct sk_buff *create_monitor_ctrl_close(struct sock *sk)
518249fa169SMarcel Holtmann {
519249fa169SMarcel Holtmann 	struct hci_mon_hdr *hdr;
520249fa169SMarcel Holtmann 	struct sk_buff *skb;
521249fa169SMarcel Holtmann 
5220ef2c42fSMarcel Holtmann 	/* No message needed when cookie is not present */
5230ef2c42fSMarcel Holtmann 	if (!hci_pi(sk)->cookie)
5240ef2c42fSMarcel Holtmann 		return NULL;
5250ef2c42fSMarcel Holtmann 
526249fa169SMarcel Holtmann 	skb = bt_skb_alloc(4, GFP_ATOMIC);
527249fa169SMarcel Holtmann 	if (!skb)
528249fa169SMarcel Holtmann 		return NULL;
529249fa169SMarcel Holtmann 
530249fa169SMarcel Holtmann 	put_unaligned_le32(hci_pi(sk)->cookie, skb_put(skb, 4));
531249fa169SMarcel Holtmann 
532249fa169SMarcel Holtmann 	__net_timestamp(skb);
533249fa169SMarcel Holtmann 
534249fa169SMarcel Holtmann 	hdr = (void *)skb_push(skb, HCI_MON_HDR_SIZE);
535249fa169SMarcel Holtmann 	hdr->opcode = cpu_to_le16(HCI_MON_CTRL_CLOSE);
5360ef2c42fSMarcel Holtmann 	if (hci_pi(sk)->hdev)
5370ef2c42fSMarcel Holtmann 		hdr->index = cpu_to_le16(hci_pi(sk)->hdev->id);
5380ef2c42fSMarcel Holtmann 	else
539249fa169SMarcel Holtmann 		hdr->index = cpu_to_le16(HCI_DEV_NONE);
540249fa169SMarcel Holtmann 	hdr->len = cpu_to_le16(skb->len - HCI_MON_HDR_SIZE);
541249fa169SMarcel Holtmann 
542249fa169SMarcel Holtmann 	return skb;
543249fa169SMarcel Holtmann }
544249fa169SMarcel Holtmann 
54538ceaa00SMarcel Holtmann static struct sk_buff *create_monitor_ctrl_command(struct sock *sk, u16 index,
54638ceaa00SMarcel Holtmann 						   u16 opcode, u16 len,
54738ceaa00SMarcel Holtmann 						   const void *buf)
54838ceaa00SMarcel Holtmann {
54938ceaa00SMarcel Holtmann 	struct hci_mon_hdr *hdr;
55038ceaa00SMarcel Holtmann 	struct sk_buff *skb;
55138ceaa00SMarcel Holtmann 
55238ceaa00SMarcel Holtmann 	skb = bt_skb_alloc(6 + len, GFP_ATOMIC);
55338ceaa00SMarcel Holtmann 	if (!skb)
55438ceaa00SMarcel Holtmann 		return NULL;
55538ceaa00SMarcel Holtmann 
55638ceaa00SMarcel Holtmann 	put_unaligned_le32(hci_pi(sk)->cookie, skb_put(skb, 4));
55738ceaa00SMarcel Holtmann 	put_unaligned_le16(opcode, skb_put(skb, 2));
55838ceaa00SMarcel Holtmann 
55938ceaa00SMarcel Holtmann 	if (buf)
56038ceaa00SMarcel Holtmann 		memcpy(skb_put(skb, len), buf, len);
56138ceaa00SMarcel Holtmann 
56238ceaa00SMarcel Holtmann 	__net_timestamp(skb);
56338ceaa00SMarcel Holtmann 
56438ceaa00SMarcel Holtmann 	hdr = (void *)skb_push(skb, HCI_MON_HDR_SIZE);
56538ceaa00SMarcel Holtmann 	hdr->opcode = cpu_to_le16(HCI_MON_CTRL_COMMAND);
56638ceaa00SMarcel Holtmann 	hdr->index = cpu_to_le16(index);
56738ceaa00SMarcel Holtmann 	hdr->len = cpu_to_le16(skb->len - HCI_MON_HDR_SIZE);
56838ceaa00SMarcel Holtmann 
56938ceaa00SMarcel Holtmann 	return skb;
57038ceaa00SMarcel Holtmann }
57138ceaa00SMarcel Holtmann 
572787b306cSJohannes Berg static void __printf(2, 3)
573787b306cSJohannes Berg send_monitor_note(struct sock *sk, const char *fmt, ...)
574dd31506dSMarcel Holtmann {
575787b306cSJohannes Berg 	size_t len;
576dd31506dSMarcel Holtmann 	struct hci_mon_hdr *hdr;
577dd31506dSMarcel Holtmann 	struct sk_buff *skb;
578787b306cSJohannes Berg 	va_list args;
579787b306cSJohannes Berg 
580787b306cSJohannes Berg 	va_start(args, fmt);
581787b306cSJohannes Berg 	len = vsnprintf(NULL, 0, fmt, args);
582787b306cSJohannes Berg 	va_end(args);
583dd31506dSMarcel Holtmann 
584dd31506dSMarcel Holtmann 	skb = bt_skb_alloc(len + 1, GFP_ATOMIC);
585dd31506dSMarcel Holtmann 	if (!skb)
586dd31506dSMarcel Holtmann 		return;
587dd31506dSMarcel Holtmann 
588787b306cSJohannes Berg 	va_start(args, fmt);
589787b306cSJohannes Berg 	vsprintf(skb_put(skb, len), fmt, args);
590787b306cSJohannes Berg 	*skb_put(skb, 1) = 0;
591787b306cSJohannes Berg 	va_end(args);
592dd31506dSMarcel Holtmann 
593dd31506dSMarcel Holtmann 	__net_timestamp(skb);
594dd31506dSMarcel Holtmann 
595dd31506dSMarcel Holtmann 	hdr = (void *)skb_push(skb, HCI_MON_HDR_SIZE);
596dd31506dSMarcel Holtmann 	hdr->opcode = cpu_to_le16(HCI_MON_SYSTEM_NOTE);
597dd31506dSMarcel Holtmann 	hdr->index = cpu_to_le16(HCI_DEV_NONE);
598dd31506dSMarcel Holtmann 	hdr->len = cpu_to_le16(skb->len - HCI_MON_HDR_SIZE);
599dd31506dSMarcel Holtmann 
600dd31506dSMarcel Holtmann 	if (sock_queue_rcv_skb(sk, skb))
601dd31506dSMarcel Holtmann 		kfree_skb(skb);
602dd31506dSMarcel Holtmann }
603dd31506dSMarcel Holtmann 
604cd82e61cSMarcel Holtmann static void send_monitor_replay(struct sock *sk)
605cd82e61cSMarcel Holtmann {
606cd82e61cSMarcel Holtmann 	struct hci_dev *hdev;
607cd82e61cSMarcel Holtmann 
608cd82e61cSMarcel Holtmann 	read_lock(&hci_dev_list_lock);
609cd82e61cSMarcel Holtmann 
610cd82e61cSMarcel Holtmann 	list_for_each_entry(hdev, &hci_dev_list, list) {
611cd82e61cSMarcel Holtmann 		struct sk_buff *skb;
612cd82e61cSMarcel Holtmann 
613cd82e61cSMarcel Holtmann 		skb = create_monitor_event(hdev, HCI_DEV_REG);
614cd82e61cSMarcel Holtmann 		if (!skb)
615cd82e61cSMarcel Holtmann 			continue;
616cd82e61cSMarcel Holtmann 
617cd82e61cSMarcel Holtmann 		if (sock_queue_rcv_skb(sk, skb))
618cd82e61cSMarcel Holtmann 			kfree_skb(skb);
61922db3cbcSMarcel Holtmann 
62022db3cbcSMarcel Holtmann 		if (!test_bit(HCI_RUNNING, &hdev->flags))
62122db3cbcSMarcel Holtmann 			continue;
62222db3cbcSMarcel Holtmann 
62322db3cbcSMarcel Holtmann 		skb = create_monitor_event(hdev, HCI_DEV_OPEN);
62422db3cbcSMarcel Holtmann 		if (!skb)
62522db3cbcSMarcel Holtmann 			continue;
62622db3cbcSMarcel Holtmann 
62722db3cbcSMarcel Holtmann 		if (sock_queue_rcv_skb(sk, skb))
62822db3cbcSMarcel Holtmann 			kfree_skb(skb);
6296c566dd5SMarcel Holtmann 
630e131d74aSMarcel Holtmann 		if (test_bit(HCI_UP, &hdev->flags))
6316c566dd5SMarcel Holtmann 			skb = create_monitor_event(hdev, HCI_DEV_UP);
632e131d74aSMarcel Holtmann 		else if (hci_dev_test_flag(hdev, HCI_SETUP))
633e131d74aSMarcel Holtmann 			skb = create_monitor_event(hdev, HCI_DEV_SETUP);
634e131d74aSMarcel Holtmann 		else
635e131d74aSMarcel Holtmann 			skb = NULL;
6366c566dd5SMarcel Holtmann 
637e131d74aSMarcel Holtmann 		if (skb) {
6386c566dd5SMarcel Holtmann 			if (sock_queue_rcv_skb(sk, skb))
6396c566dd5SMarcel Holtmann 				kfree_skb(skb);
640cd82e61cSMarcel Holtmann 		}
641e131d74aSMarcel Holtmann 	}
642cd82e61cSMarcel Holtmann 
643cd82e61cSMarcel Holtmann 	read_unlock(&hci_dev_list_lock);
644cd82e61cSMarcel Holtmann }
645cd82e61cSMarcel Holtmann 
646249fa169SMarcel Holtmann static void send_monitor_control_replay(struct sock *mon_sk)
647249fa169SMarcel Holtmann {
648249fa169SMarcel Holtmann 	struct sock *sk;
649249fa169SMarcel Holtmann 
650249fa169SMarcel Holtmann 	read_lock(&hci_sk_list.lock);
651249fa169SMarcel Holtmann 
652249fa169SMarcel Holtmann 	sk_for_each(sk, &hci_sk_list.head) {
653249fa169SMarcel Holtmann 		struct sk_buff *skb;
654249fa169SMarcel Holtmann 
655249fa169SMarcel Holtmann 		if (hci_pi(sk)->channel != HCI_CHANNEL_CONTROL)
656249fa169SMarcel Holtmann 			continue;
657249fa169SMarcel Holtmann 
658249fa169SMarcel Holtmann 		skb = create_monitor_ctrl_open(sk);
659249fa169SMarcel Holtmann 		if (!skb)
660249fa169SMarcel Holtmann 			continue;
661249fa169SMarcel Holtmann 
662249fa169SMarcel Holtmann 		if (sock_queue_rcv_skb(mon_sk, skb))
663249fa169SMarcel Holtmann 			kfree_skb(skb);
664249fa169SMarcel Holtmann 	}
665249fa169SMarcel Holtmann 
666249fa169SMarcel Holtmann 	read_unlock(&hci_sk_list.lock);
667249fa169SMarcel Holtmann }
668249fa169SMarcel Holtmann 
669040030efSMarcel Holtmann /* Generate internal stack event */
670040030efSMarcel Holtmann static void hci_si_event(struct hci_dev *hdev, int type, int dlen, void *data)
671040030efSMarcel Holtmann {
672040030efSMarcel Holtmann 	struct hci_event_hdr *hdr;
673040030efSMarcel Holtmann 	struct hci_ev_stack_internal *ev;
674040030efSMarcel Holtmann 	struct sk_buff *skb;
675040030efSMarcel Holtmann 
676040030efSMarcel Holtmann 	skb = bt_skb_alloc(HCI_EVENT_HDR_SIZE + sizeof(*ev) + dlen, GFP_ATOMIC);
677040030efSMarcel Holtmann 	if (!skb)
678040030efSMarcel Holtmann 		return;
679040030efSMarcel Holtmann 
680040030efSMarcel Holtmann 	hdr = (void *)skb_put(skb, HCI_EVENT_HDR_SIZE);
681040030efSMarcel Holtmann 	hdr->evt  = HCI_EV_STACK_INTERNAL;
682040030efSMarcel Holtmann 	hdr->plen = sizeof(*ev) + dlen;
683040030efSMarcel Holtmann 
684040030efSMarcel Holtmann 	ev  = (void *)skb_put(skb, sizeof(*ev) + dlen);
685040030efSMarcel Holtmann 	ev->type = type;
686040030efSMarcel Holtmann 	memcpy(ev->data, data, dlen);
687040030efSMarcel Holtmann 
688040030efSMarcel Holtmann 	bt_cb(skb)->incoming = 1;
689040030efSMarcel Holtmann 	__net_timestamp(skb);
690040030efSMarcel Holtmann 
691d79f34e3SMarcel Holtmann 	hci_skb_pkt_type(skb) = HCI_EVENT_PKT;
692040030efSMarcel Holtmann 	hci_send_to_sock(hdev, skb);
693040030efSMarcel Holtmann 	kfree_skb(skb);
694040030efSMarcel Holtmann }
695040030efSMarcel Holtmann 
696040030efSMarcel Holtmann void hci_sock_dev_event(struct hci_dev *hdev, int event)
697040030efSMarcel Holtmann {
698040030efSMarcel Holtmann 	BT_DBG("hdev %s event %d", hdev->name, event);
699040030efSMarcel Holtmann 
700cd82e61cSMarcel Holtmann 	if (atomic_read(&monitor_promisc)) {
701cd82e61cSMarcel Holtmann 		struct sk_buff *skb;
702cd82e61cSMarcel Holtmann 
703ed1b28a4SMarcel Holtmann 		/* Send event to monitor */
704cd82e61cSMarcel Holtmann 		skb = create_monitor_event(hdev, event);
705cd82e61cSMarcel Holtmann 		if (skb) {
706c08b1a1dSMarcel Holtmann 			hci_send_to_channel(HCI_CHANNEL_MONITOR, skb,
707c08b1a1dSMarcel Holtmann 					    HCI_SOCK_TRUSTED, NULL);
708cd82e61cSMarcel Holtmann 			kfree_skb(skb);
709cd82e61cSMarcel Holtmann 		}
710cd82e61cSMarcel Holtmann 	}
711cd82e61cSMarcel Holtmann 
712ed1b28a4SMarcel Holtmann 	if (event <= HCI_DEV_DOWN) {
713ed1b28a4SMarcel Holtmann 		struct hci_ev_si_device ev;
714ed1b28a4SMarcel Holtmann 
715040030efSMarcel Holtmann 		/* Send event to sockets */
716040030efSMarcel Holtmann 		ev.event  = event;
717040030efSMarcel Holtmann 		ev.dev_id = hdev->id;
718040030efSMarcel Holtmann 		hci_si_event(NULL, HCI_EV_SI_DEVICE, sizeof(ev), &ev);
719ed1b28a4SMarcel Holtmann 	}
720040030efSMarcel Holtmann 
721040030efSMarcel Holtmann 	if (event == HCI_DEV_UNREG) {
722040030efSMarcel Holtmann 		struct sock *sk;
723040030efSMarcel Holtmann 
724040030efSMarcel Holtmann 		/* Detach sockets from device */
725040030efSMarcel Holtmann 		read_lock(&hci_sk_list.lock);
726b67bfe0dSSasha Levin 		sk_for_each(sk, &hci_sk_list.head) {
727040030efSMarcel Holtmann 			bh_lock_sock_nested(sk);
728040030efSMarcel Holtmann 			if (hci_pi(sk)->hdev == hdev) {
729040030efSMarcel Holtmann 				hci_pi(sk)->hdev = NULL;
730040030efSMarcel Holtmann 				sk->sk_err = EPIPE;
731040030efSMarcel Holtmann 				sk->sk_state = BT_OPEN;
732040030efSMarcel Holtmann 				sk->sk_state_change(sk);
733040030efSMarcel Holtmann 
734040030efSMarcel Holtmann 				hci_dev_put(hdev);
735040030efSMarcel Holtmann 			}
736040030efSMarcel Holtmann 			bh_unlock_sock(sk);
737040030efSMarcel Holtmann 		}
738040030efSMarcel Holtmann 		read_unlock(&hci_sk_list.lock);
739040030efSMarcel Holtmann 	}
740040030efSMarcel Holtmann }
741040030efSMarcel Holtmann 
742801c1e8dSJohan Hedberg static struct hci_mgmt_chan *__hci_mgmt_chan_find(unsigned short channel)
743801c1e8dSJohan Hedberg {
744801c1e8dSJohan Hedberg 	struct hci_mgmt_chan *c;
745801c1e8dSJohan Hedberg 
746801c1e8dSJohan Hedberg 	list_for_each_entry(c, &mgmt_chan_list, list) {
747801c1e8dSJohan Hedberg 		if (c->channel == channel)
748801c1e8dSJohan Hedberg 			return c;
749801c1e8dSJohan Hedberg 	}
750801c1e8dSJohan Hedberg 
751801c1e8dSJohan Hedberg 	return NULL;
752801c1e8dSJohan Hedberg }
753801c1e8dSJohan Hedberg 
754801c1e8dSJohan Hedberg static struct hci_mgmt_chan *hci_mgmt_chan_find(unsigned short channel)
755801c1e8dSJohan Hedberg {
756801c1e8dSJohan Hedberg 	struct hci_mgmt_chan *c;
757801c1e8dSJohan Hedberg 
758801c1e8dSJohan Hedberg 	mutex_lock(&mgmt_chan_list_lock);
759801c1e8dSJohan Hedberg 	c = __hci_mgmt_chan_find(channel);
760801c1e8dSJohan Hedberg 	mutex_unlock(&mgmt_chan_list_lock);
761801c1e8dSJohan Hedberg 
762801c1e8dSJohan Hedberg 	return c;
763801c1e8dSJohan Hedberg }
764801c1e8dSJohan Hedberg 
765801c1e8dSJohan Hedberg int hci_mgmt_chan_register(struct hci_mgmt_chan *c)
766801c1e8dSJohan Hedberg {
767801c1e8dSJohan Hedberg 	if (c->channel < HCI_CHANNEL_CONTROL)
768801c1e8dSJohan Hedberg 		return -EINVAL;
769801c1e8dSJohan Hedberg 
770801c1e8dSJohan Hedberg 	mutex_lock(&mgmt_chan_list_lock);
771801c1e8dSJohan Hedberg 	if (__hci_mgmt_chan_find(c->channel)) {
772801c1e8dSJohan Hedberg 		mutex_unlock(&mgmt_chan_list_lock);
773801c1e8dSJohan Hedberg 		return -EALREADY;
774801c1e8dSJohan Hedberg 	}
775801c1e8dSJohan Hedberg 
776801c1e8dSJohan Hedberg 	list_add_tail(&c->list, &mgmt_chan_list);
777801c1e8dSJohan Hedberg 
778801c1e8dSJohan Hedberg 	mutex_unlock(&mgmt_chan_list_lock);
779801c1e8dSJohan Hedberg 
780801c1e8dSJohan Hedberg 	return 0;
781801c1e8dSJohan Hedberg }
782801c1e8dSJohan Hedberg EXPORT_SYMBOL(hci_mgmt_chan_register);
783801c1e8dSJohan Hedberg 
784801c1e8dSJohan Hedberg void hci_mgmt_chan_unregister(struct hci_mgmt_chan *c)
785801c1e8dSJohan Hedberg {
786801c1e8dSJohan Hedberg 	mutex_lock(&mgmt_chan_list_lock);
787801c1e8dSJohan Hedberg 	list_del(&c->list);
788801c1e8dSJohan Hedberg 	mutex_unlock(&mgmt_chan_list_lock);
789801c1e8dSJohan Hedberg }
790801c1e8dSJohan Hedberg EXPORT_SYMBOL(hci_mgmt_chan_unregister);
791801c1e8dSJohan Hedberg 
7921da177e4SLinus Torvalds static int hci_sock_release(struct socket *sock)
7931da177e4SLinus Torvalds {
7941da177e4SLinus Torvalds 	struct sock *sk = sock->sk;
7957b005bd3SMarcel Holtmann 	struct hci_dev *hdev;
796249fa169SMarcel Holtmann 	struct sk_buff *skb;
7971da177e4SLinus Torvalds 
7981da177e4SLinus Torvalds 	BT_DBG("sock %p sk %p", sock, sk);
7991da177e4SLinus Torvalds 
8001da177e4SLinus Torvalds 	if (!sk)
8011da177e4SLinus Torvalds 		return 0;
8021da177e4SLinus Torvalds 
8037b005bd3SMarcel Holtmann 	hdev = hci_pi(sk)->hdev;
8047b005bd3SMarcel Holtmann 
80570ecce91SMarcel Holtmann 	switch (hci_pi(sk)->channel) {
80670ecce91SMarcel Holtmann 	case HCI_CHANNEL_MONITOR:
807cd82e61cSMarcel Holtmann 		atomic_dec(&monitor_promisc);
80870ecce91SMarcel Holtmann 		break;
80970ecce91SMarcel Holtmann 	case HCI_CHANNEL_CONTROL:
810249fa169SMarcel Holtmann 		/* Send event to monitor */
811249fa169SMarcel Holtmann 		skb = create_monitor_ctrl_close(sk);
812249fa169SMarcel Holtmann 		if (skb) {
813249fa169SMarcel Holtmann 			hci_send_to_channel(HCI_CHANNEL_MONITOR, skb,
814249fa169SMarcel Holtmann 					    HCI_SOCK_TRUSTED, NULL);
815249fa169SMarcel Holtmann 			kfree_skb(skb);
816249fa169SMarcel Holtmann 		}
817249fa169SMarcel Holtmann 
818df1cb87aSMarcel Holtmann 		hci_sock_free_cookie(sk);
81970ecce91SMarcel Holtmann 		break;
82070ecce91SMarcel Holtmann 	}
821cd82e61cSMarcel Holtmann 
8221da177e4SLinus Torvalds 	bt_sock_unlink(&hci_sk_list, sk);
8231da177e4SLinus Torvalds 
8241da177e4SLinus Torvalds 	if (hdev) {
82523500189SMarcel Holtmann 		if (hci_pi(sk)->channel == HCI_CHANNEL_USER) {
8266b3cc1dbSSimon Fels 			/* When releasing an user channel exclusive access,
8276b3cc1dbSSimon Fels 			 * call hci_dev_do_close directly instead of calling
8286b3cc1dbSSimon Fels 			 * hci_dev_close to ensure the exclusive access will
8296b3cc1dbSSimon Fels 			 * be released and the controller brought back down.
8306b3cc1dbSSimon Fels 			 *
8316b3cc1dbSSimon Fels 			 * The checking of HCI_AUTO_OFF is not needed in this
8326b3cc1dbSSimon Fels 			 * case since it will have been cleared already when
8336b3cc1dbSSimon Fels 			 * opening the user channel.
8346b3cc1dbSSimon Fels 			 */
8356b3cc1dbSSimon Fels 			hci_dev_do_close(hdev);
8369380f9eaSLoic Poulain 			hci_dev_clear_flag(hdev, HCI_USER_CHANNEL);
8379380f9eaSLoic Poulain 			mgmt_index_added(hdev);
83823500189SMarcel Holtmann 		}
83923500189SMarcel Holtmann 
8401da177e4SLinus Torvalds 		atomic_dec(&hdev->promisc);
8411da177e4SLinus Torvalds 		hci_dev_put(hdev);
8421da177e4SLinus Torvalds 	}
8431da177e4SLinus Torvalds 
8441da177e4SLinus Torvalds 	sock_orphan(sk);
8451da177e4SLinus Torvalds 
8461da177e4SLinus Torvalds 	skb_queue_purge(&sk->sk_receive_queue);
8471da177e4SLinus Torvalds 	skb_queue_purge(&sk->sk_write_queue);
8481da177e4SLinus Torvalds 
8491da177e4SLinus Torvalds 	sock_put(sk);
8501da177e4SLinus Torvalds 	return 0;
8511da177e4SLinus Torvalds }
8521da177e4SLinus Torvalds 
853b2a66aadSAntti Julku static int hci_sock_blacklist_add(struct hci_dev *hdev, void __user *arg)
854f0358568SJohan Hedberg {
855f0358568SJohan Hedberg 	bdaddr_t bdaddr;
8565e762444SAntti Julku 	int err;
857f0358568SJohan Hedberg 
858f0358568SJohan Hedberg 	if (copy_from_user(&bdaddr, arg, sizeof(bdaddr)))
859f0358568SJohan Hedberg 		return -EFAULT;
860f0358568SJohan Hedberg 
86109fd0de5SGustavo F. Padovan 	hci_dev_lock(hdev);
8625e762444SAntti Julku 
863dcc36c16SJohan Hedberg 	err = hci_bdaddr_list_add(&hdev->blacklist, &bdaddr, BDADDR_BREDR);
8645e762444SAntti Julku 
86509fd0de5SGustavo F. Padovan 	hci_dev_unlock(hdev);
8665e762444SAntti Julku 
8675e762444SAntti Julku 	return err;
868f0358568SJohan Hedberg }
869f0358568SJohan Hedberg 
870b2a66aadSAntti Julku static int hci_sock_blacklist_del(struct hci_dev *hdev, void __user *arg)
871f0358568SJohan Hedberg {
872f0358568SJohan Hedberg 	bdaddr_t bdaddr;
8735e762444SAntti Julku 	int err;
874f0358568SJohan Hedberg 
875f0358568SJohan Hedberg 	if (copy_from_user(&bdaddr, arg, sizeof(bdaddr)))
876f0358568SJohan Hedberg 		return -EFAULT;
877f0358568SJohan Hedberg 
87809fd0de5SGustavo F. Padovan 	hci_dev_lock(hdev);
8795e762444SAntti Julku 
880dcc36c16SJohan Hedberg 	err = hci_bdaddr_list_del(&hdev->blacklist, &bdaddr, BDADDR_BREDR);
8815e762444SAntti Julku 
88209fd0de5SGustavo F. Padovan 	hci_dev_unlock(hdev);
8835e762444SAntti Julku 
8845e762444SAntti Julku 	return err;
885f0358568SJohan Hedberg }
886f0358568SJohan Hedberg 
8871da177e4SLinus Torvalds /* Ioctls that require bound socket */
8886039aa73SGustavo Padovan static int hci_sock_bound_ioctl(struct sock *sk, unsigned int cmd,
8896039aa73SGustavo Padovan 				unsigned long arg)
8901da177e4SLinus Torvalds {
8911da177e4SLinus Torvalds 	struct hci_dev *hdev = hci_pi(sk)->hdev;
8921da177e4SLinus Torvalds 
8931da177e4SLinus Torvalds 	if (!hdev)
8941da177e4SLinus Torvalds 		return -EBADFD;
8951da177e4SLinus Torvalds 
896d7a5a11dSMarcel Holtmann 	if (hci_dev_test_flag(hdev, HCI_USER_CHANNEL))
8970736cfa8SMarcel Holtmann 		return -EBUSY;
8980736cfa8SMarcel Holtmann 
899d7a5a11dSMarcel Holtmann 	if (hci_dev_test_flag(hdev, HCI_UNCONFIGURED))
900fee746b0SMarcel Holtmann 		return -EOPNOTSUPP;
901fee746b0SMarcel Holtmann 
902ca8bee5dSMarcel Holtmann 	if (hdev->dev_type != HCI_PRIMARY)
9035b69bef5SMarcel Holtmann 		return -EOPNOTSUPP;
9045b69bef5SMarcel Holtmann 
9051da177e4SLinus Torvalds 	switch (cmd) {
9061da177e4SLinus Torvalds 	case HCISETRAW:
9071da177e4SLinus Torvalds 		if (!capable(CAP_NET_ADMIN))
908bf5b30b8SZhao Hongjiang 			return -EPERM;
909db596681SMarcel Holtmann 		return -EOPNOTSUPP;
9101da177e4SLinus Torvalds 
9111da177e4SLinus Torvalds 	case HCIGETCONNINFO:
9121da177e4SLinus Torvalds 		return hci_get_conn_info(hdev, (void __user *)arg);
9131da177e4SLinus Torvalds 
91440be492fSMarcel Holtmann 	case HCIGETAUTHINFO:
91540be492fSMarcel Holtmann 		return hci_get_auth_info(hdev, (void __user *)arg);
91640be492fSMarcel Holtmann 
917f0358568SJohan Hedberg 	case HCIBLOCKADDR:
918f0358568SJohan Hedberg 		if (!capable(CAP_NET_ADMIN))
919bf5b30b8SZhao Hongjiang 			return -EPERM;
920b2a66aadSAntti Julku 		return hci_sock_blacklist_add(hdev, (void __user *)arg);
921f0358568SJohan Hedberg 
922f0358568SJohan Hedberg 	case HCIUNBLOCKADDR:
923f0358568SJohan Hedberg 		if (!capable(CAP_NET_ADMIN))
924bf5b30b8SZhao Hongjiang 			return -EPERM;
925b2a66aadSAntti Julku 		return hci_sock_blacklist_del(hdev, (void __user *)arg);
9260736cfa8SMarcel Holtmann 	}
927f0358568SJohan Hedberg 
928324d36edSMarcel Holtmann 	return -ENOIOCTLCMD;
9291da177e4SLinus Torvalds }
9301da177e4SLinus Torvalds 
9318fc9ced3SGustavo Padovan static int hci_sock_ioctl(struct socket *sock, unsigned int cmd,
9328fc9ced3SGustavo Padovan 			  unsigned long arg)
9331da177e4SLinus Torvalds {
9341da177e4SLinus Torvalds 	void __user *argp = (void __user *)arg;
9350736cfa8SMarcel Holtmann 	struct sock *sk = sock->sk;
9361da177e4SLinus Torvalds 	int err;
9371da177e4SLinus Torvalds 
9381da177e4SLinus Torvalds 	BT_DBG("cmd %x arg %lx", cmd, arg);
9391da177e4SLinus Torvalds 
940c1c4f956SMarcel Holtmann 	lock_sock(sk);
941c1c4f956SMarcel Holtmann 
942c1c4f956SMarcel Holtmann 	if (hci_pi(sk)->channel != HCI_CHANNEL_RAW) {
943c1c4f956SMarcel Holtmann 		err = -EBADFD;
944c1c4f956SMarcel Holtmann 		goto done;
945c1c4f956SMarcel Holtmann 	}
946c1c4f956SMarcel Holtmann 
947c1c4f956SMarcel Holtmann 	release_sock(sk);
948c1c4f956SMarcel Holtmann 
9491da177e4SLinus Torvalds 	switch (cmd) {
9501da177e4SLinus Torvalds 	case HCIGETDEVLIST:
9511da177e4SLinus Torvalds 		return hci_get_dev_list(argp);
9521da177e4SLinus Torvalds 
9531da177e4SLinus Torvalds 	case HCIGETDEVINFO:
9541da177e4SLinus Torvalds 		return hci_get_dev_info(argp);
9551da177e4SLinus Torvalds 
9561da177e4SLinus Torvalds 	case HCIGETCONNLIST:
9571da177e4SLinus Torvalds 		return hci_get_conn_list(argp);
9581da177e4SLinus Torvalds 
9591da177e4SLinus Torvalds 	case HCIDEVUP:
9601da177e4SLinus Torvalds 		if (!capable(CAP_NET_ADMIN))
961bf5b30b8SZhao Hongjiang 			return -EPERM;
9621da177e4SLinus Torvalds 		return hci_dev_open(arg);
9631da177e4SLinus Torvalds 
9641da177e4SLinus Torvalds 	case HCIDEVDOWN:
9651da177e4SLinus Torvalds 		if (!capable(CAP_NET_ADMIN))
966bf5b30b8SZhao Hongjiang 			return -EPERM;
9671da177e4SLinus Torvalds 		return hci_dev_close(arg);
9681da177e4SLinus Torvalds 
9691da177e4SLinus Torvalds 	case HCIDEVRESET:
9701da177e4SLinus Torvalds 		if (!capable(CAP_NET_ADMIN))
971bf5b30b8SZhao Hongjiang 			return -EPERM;
9721da177e4SLinus Torvalds 		return hci_dev_reset(arg);
9731da177e4SLinus Torvalds 
9741da177e4SLinus Torvalds 	case HCIDEVRESTAT:
9751da177e4SLinus Torvalds 		if (!capable(CAP_NET_ADMIN))
976bf5b30b8SZhao Hongjiang 			return -EPERM;
9771da177e4SLinus Torvalds 		return hci_dev_reset_stat(arg);
9781da177e4SLinus Torvalds 
9791da177e4SLinus Torvalds 	case HCISETSCAN:
9801da177e4SLinus Torvalds 	case HCISETAUTH:
9811da177e4SLinus Torvalds 	case HCISETENCRYPT:
9821da177e4SLinus Torvalds 	case HCISETPTYPE:
9831da177e4SLinus Torvalds 	case HCISETLINKPOL:
9841da177e4SLinus Torvalds 	case HCISETLINKMODE:
9851da177e4SLinus Torvalds 	case HCISETACLMTU:
9861da177e4SLinus Torvalds 	case HCISETSCOMTU:
9871da177e4SLinus Torvalds 		if (!capable(CAP_NET_ADMIN))
988bf5b30b8SZhao Hongjiang 			return -EPERM;
9891da177e4SLinus Torvalds 		return hci_dev_cmd(cmd, argp);
9901da177e4SLinus Torvalds 
9911da177e4SLinus Torvalds 	case HCIINQUIRY:
9921da177e4SLinus Torvalds 		return hci_inquiry(argp);
993c1c4f956SMarcel Holtmann 	}
9941da177e4SLinus Torvalds 
9951da177e4SLinus Torvalds 	lock_sock(sk);
996c1c4f956SMarcel Holtmann 
9971da177e4SLinus Torvalds 	err = hci_sock_bound_ioctl(sk, cmd, arg);
998c1c4f956SMarcel Holtmann 
999c1c4f956SMarcel Holtmann done:
10001da177e4SLinus Torvalds 	release_sock(sk);
10011da177e4SLinus Torvalds 	return err;
10021da177e4SLinus Torvalds }
10031da177e4SLinus Torvalds 
10048fc9ced3SGustavo Padovan static int hci_sock_bind(struct socket *sock, struct sockaddr *addr,
10058fc9ced3SGustavo Padovan 			 int addr_len)
10061da177e4SLinus Torvalds {
10070381101fSJohan Hedberg 	struct sockaddr_hci haddr;
10081da177e4SLinus Torvalds 	struct sock *sk = sock->sk;
10091da177e4SLinus Torvalds 	struct hci_dev *hdev = NULL;
10100381101fSJohan Hedberg 	int len, err = 0;
10111da177e4SLinus Torvalds 
10121da177e4SLinus Torvalds 	BT_DBG("sock %p sk %p", sock, sk);
10131da177e4SLinus Torvalds 
10140381101fSJohan Hedberg 	if (!addr)
10150381101fSJohan Hedberg 		return -EINVAL;
10160381101fSJohan Hedberg 
10170381101fSJohan Hedberg 	memset(&haddr, 0, sizeof(haddr));
10180381101fSJohan Hedberg 	len = min_t(unsigned int, sizeof(haddr), addr_len);
10190381101fSJohan Hedberg 	memcpy(&haddr, addr, len);
10200381101fSJohan Hedberg 
10210381101fSJohan Hedberg 	if (haddr.hci_family != AF_BLUETOOTH)
10220381101fSJohan Hedberg 		return -EINVAL;
10230381101fSJohan Hedberg 
10241da177e4SLinus Torvalds 	lock_sock(sk);
10251da177e4SLinus Torvalds 
10267cc2ade2SMarcel Holtmann 	if (sk->sk_state == BT_BOUND) {
10277cc2ade2SMarcel Holtmann 		err = -EALREADY;
10287cc2ade2SMarcel Holtmann 		goto done;
10297cc2ade2SMarcel Holtmann 	}
10307cc2ade2SMarcel Holtmann 
10317cc2ade2SMarcel Holtmann 	switch (haddr.hci_channel) {
10327cc2ade2SMarcel Holtmann 	case HCI_CHANNEL_RAW:
10337cc2ade2SMarcel Holtmann 		if (hci_pi(sk)->hdev) {
10341da177e4SLinus Torvalds 			err = -EALREADY;
10351da177e4SLinus Torvalds 			goto done;
10361da177e4SLinus Torvalds 		}
10371da177e4SLinus Torvalds 
10380381101fSJohan Hedberg 		if (haddr.hci_dev != HCI_DEV_NONE) {
10390381101fSJohan Hedberg 			hdev = hci_dev_get(haddr.hci_dev);
104070f23020SAndrei Emeltchenko 			if (!hdev) {
10411da177e4SLinus Torvalds 				err = -ENODEV;
10421da177e4SLinus Torvalds 				goto done;
10431da177e4SLinus Torvalds 			}
10441da177e4SLinus Torvalds 
10451da177e4SLinus Torvalds 			atomic_inc(&hdev->promisc);
10461da177e4SLinus Torvalds 		}
10471da177e4SLinus Torvalds 
1048*5a6d2cf5SMarcel Holtmann 		hci_pi(sk)->channel = haddr.hci_channel;
10491da177e4SLinus Torvalds 		hci_pi(sk)->hdev = hdev;
10507cc2ade2SMarcel Holtmann 		break;
10517cc2ade2SMarcel Holtmann 
105223500189SMarcel Holtmann 	case HCI_CHANNEL_USER:
105323500189SMarcel Holtmann 		if (hci_pi(sk)->hdev) {
105423500189SMarcel Holtmann 			err = -EALREADY;
105523500189SMarcel Holtmann 			goto done;
105623500189SMarcel Holtmann 		}
105723500189SMarcel Holtmann 
105823500189SMarcel Holtmann 		if (haddr.hci_dev == HCI_DEV_NONE) {
105923500189SMarcel Holtmann 			err = -EINVAL;
106023500189SMarcel Holtmann 			goto done;
106123500189SMarcel Holtmann 		}
106223500189SMarcel Holtmann 
106310a8b86fSMarcel Holtmann 		if (!capable(CAP_NET_ADMIN)) {
106423500189SMarcel Holtmann 			err = -EPERM;
106523500189SMarcel Holtmann 			goto done;
106623500189SMarcel Holtmann 		}
106723500189SMarcel Holtmann 
106823500189SMarcel Holtmann 		hdev = hci_dev_get(haddr.hci_dev);
106923500189SMarcel Holtmann 		if (!hdev) {
107023500189SMarcel Holtmann 			err = -ENODEV;
107123500189SMarcel Holtmann 			goto done;
107223500189SMarcel Holtmann 		}
107323500189SMarcel Holtmann 
1074781f899fSMarcel Holtmann 		if (test_bit(HCI_INIT, &hdev->flags) ||
1075d7a5a11dSMarcel Holtmann 		    hci_dev_test_flag(hdev, HCI_SETUP) ||
1076781f899fSMarcel Holtmann 		    hci_dev_test_flag(hdev, HCI_CONFIG) ||
1077781f899fSMarcel Holtmann 		    (!hci_dev_test_flag(hdev, HCI_AUTO_OFF) &&
1078781f899fSMarcel Holtmann 		     test_bit(HCI_UP, &hdev->flags))) {
107923500189SMarcel Holtmann 			err = -EBUSY;
108023500189SMarcel Holtmann 			hci_dev_put(hdev);
108123500189SMarcel Holtmann 			goto done;
108223500189SMarcel Holtmann 		}
108323500189SMarcel Holtmann 
1084238be788SMarcel Holtmann 		if (hci_dev_test_and_set_flag(hdev, HCI_USER_CHANNEL)) {
108523500189SMarcel Holtmann 			err = -EUSERS;
108623500189SMarcel Holtmann 			hci_dev_put(hdev);
108723500189SMarcel Holtmann 			goto done;
108823500189SMarcel Holtmann 		}
108923500189SMarcel Holtmann 
109023500189SMarcel Holtmann 		mgmt_index_removed(hdev);
109123500189SMarcel Holtmann 
109223500189SMarcel Holtmann 		err = hci_dev_open(hdev->id);
109323500189SMarcel Holtmann 		if (err) {
1094781f899fSMarcel Holtmann 			if (err == -EALREADY) {
1095781f899fSMarcel Holtmann 				/* In case the transport is already up and
1096781f899fSMarcel Holtmann 				 * running, clear the error here.
1097781f899fSMarcel Holtmann 				 *
1098781f899fSMarcel Holtmann 				 * This can happen when opening an user
1099781f899fSMarcel Holtmann 				 * channel and HCI_AUTO_OFF grace period
1100781f899fSMarcel Holtmann 				 * is still active.
1101781f899fSMarcel Holtmann 				 */
1102781f899fSMarcel Holtmann 				err = 0;
1103781f899fSMarcel Holtmann 			} else {
1104a358dc11SMarcel Holtmann 				hci_dev_clear_flag(hdev, HCI_USER_CHANNEL);
1105c6521401SMarcel Holtmann 				mgmt_index_added(hdev);
110623500189SMarcel Holtmann 				hci_dev_put(hdev);
110723500189SMarcel Holtmann 				goto done;
110823500189SMarcel Holtmann 			}
1109781f899fSMarcel Holtmann 		}
111023500189SMarcel Holtmann 
1111*5a6d2cf5SMarcel Holtmann 		hci_pi(sk)->channel = haddr.hci_channel;
111223500189SMarcel Holtmann 		hci_pi(sk)->hdev = hdev;
1113*5a6d2cf5SMarcel Holtmann 
1114*5a6d2cf5SMarcel Holtmann 		atomic_inc(&hdev->promisc);
111523500189SMarcel Holtmann 		break;
111623500189SMarcel Holtmann 
1117cd82e61cSMarcel Holtmann 	case HCI_CHANNEL_MONITOR:
1118cd82e61cSMarcel Holtmann 		if (haddr.hci_dev != HCI_DEV_NONE) {
1119cd82e61cSMarcel Holtmann 			err = -EINVAL;
1120cd82e61cSMarcel Holtmann 			goto done;
1121cd82e61cSMarcel Holtmann 		}
1122cd82e61cSMarcel Holtmann 
1123cd82e61cSMarcel Holtmann 		if (!capable(CAP_NET_RAW)) {
1124cd82e61cSMarcel Holtmann 			err = -EPERM;
1125cd82e61cSMarcel Holtmann 			goto done;
1126cd82e61cSMarcel Holtmann 		}
1127cd82e61cSMarcel Holtmann 
1128*5a6d2cf5SMarcel Holtmann 		hci_pi(sk)->channel = haddr.hci_channel;
1129*5a6d2cf5SMarcel Holtmann 
113050ebc055SMarcel Holtmann 		/* The monitor interface is restricted to CAP_NET_RAW
113150ebc055SMarcel Holtmann 		 * capabilities and with that implicitly trusted.
113250ebc055SMarcel Holtmann 		 */
113350ebc055SMarcel Holtmann 		hci_sock_set_flag(sk, HCI_SOCK_TRUSTED);
113450ebc055SMarcel Holtmann 
1135787b306cSJohannes Berg 		send_monitor_note(sk, "Linux version %s (%s)",
1136787b306cSJohannes Berg 				  init_utsname()->release,
1137787b306cSJohannes Berg 				  init_utsname()->machine);
11389e8305b3SMarcel Holtmann 		send_monitor_note(sk, "Bluetooth subsystem version %u.%u",
11399e8305b3SMarcel Holtmann 				  BT_SUBSYS_VERSION, BT_SUBSYS_REVISION);
1140cd82e61cSMarcel Holtmann 		send_monitor_replay(sk);
1141249fa169SMarcel Holtmann 		send_monitor_control_replay(sk);
1142cd82e61cSMarcel Holtmann 
1143cd82e61cSMarcel Holtmann 		atomic_inc(&monitor_promisc);
1144cd82e61cSMarcel Holtmann 		break;
1145cd82e61cSMarcel Holtmann 
1146ac714949SMarcel Holtmann 	case HCI_CHANNEL_LOGGING:
1147ac714949SMarcel Holtmann 		if (haddr.hci_dev != HCI_DEV_NONE) {
1148ac714949SMarcel Holtmann 			err = -EINVAL;
1149ac714949SMarcel Holtmann 			goto done;
1150ac714949SMarcel Holtmann 		}
1151ac714949SMarcel Holtmann 
1152ac714949SMarcel Holtmann 		if (!capable(CAP_NET_ADMIN)) {
1153ac714949SMarcel Holtmann 			err = -EPERM;
1154ac714949SMarcel Holtmann 			goto done;
1155ac714949SMarcel Holtmann 		}
1156*5a6d2cf5SMarcel Holtmann 
1157*5a6d2cf5SMarcel Holtmann 		hci_pi(sk)->channel = haddr.hci_channel;
1158ac714949SMarcel Holtmann 		break;
1159ac714949SMarcel Holtmann 
11607cc2ade2SMarcel Holtmann 	default:
1161801c1e8dSJohan Hedberg 		if (!hci_mgmt_chan_find(haddr.hci_channel)) {
11627cc2ade2SMarcel Holtmann 			err = -EINVAL;
11637cc2ade2SMarcel Holtmann 			goto done;
11647cc2ade2SMarcel Holtmann 		}
11657cc2ade2SMarcel Holtmann 
1166801c1e8dSJohan Hedberg 		if (haddr.hci_dev != HCI_DEV_NONE) {
1167801c1e8dSJohan Hedberg 			err = -EINVAL;
1168801c1e8dSJohan Hedberg 			goto done;
1169801c1e8dSJohan Hedberg 		}
1170801c1e8dSJohan Hedberg 
11711195fbb8SMarcel Holtmann 		/* Users with CAP_NET_ADMIN capabilities are allowed
11721195fbb8SMarcel Holtmann 		 * access to all management commands and events. For
11731195fbb8SMarcel Holtmann 		 * untrusted users the interface is restricted and
11741195fbb8SMarcel Holtmann 		 * also only untrusted events are sent.
117550ebc055SMarcel Holtmann 		 */
11761195fbb8SMarcel Holtmann 		if (capable(CAP_NET_ADMIN))
117750ebc055SMarcel Holtmann 			hci_sock_set_flag(sk, HCI_SOCK_TRUSTED);
117850ebc055SMarcel Holtmann 
1179*5a6d2cf5SMarcel Holtmann 		hci_pi(sk)->channel = haddr.hci_channel;
1180*5a6d2cf5SMarcel Holtmann 
1181f9207338SMarcel Holtmann 		/* At the moment the index and unconfigured index events
1182f9207338SMarcel Holtmann 		 * are enabled unconditionally. Setting them on each
1183f9207338SMarcel Holtmann 		 * socket when binding keeps this functionality. They
1184f9207338SMarcel Holtmann 		 * however might be cleared later and then sending of these
1185f9207338SMarcel Holtmann 		 * events will be disabled, but that is then intentional.
1186f6b7712eSMarcel Holtmann 		 *
1187f6b7712eSMarcel Holtmann 		 * This also enables generic events that are safe to be
1188f6b7712eSMarcel Holtmann 		 * received by untrusted users. Example for such events
1189f6b7712eSMarcel Holtmann 		 * are changes to settings, class of device, name etc.
1190f9207338SMarcel Holtmann 		 */
1191*5a6d2cf5SMarcel Holtmann 		if (hci_pi(sk)->channel == HCI_CHANNEL_CONTROL) {
1192249fa169SMarcel Holtmann 			struct sk_buff *skb;
119370ecce91SMarcel Holtmann 
1194df1cb87aSMarcel Holtmann 			hci_sock_gen_cookie(sk);
119570ecce91SMarcel Holtmann 
1196249fa169SMarcel Holtmann 			/* Send event to monitor */
1197249fa169SMarcel Holtmann 			skb = create_monitor_ctrl_open(sk);
1198249fa169SMarcel Holtmann 			if (skb) {
1199249fa169SMarcel Holtmann 				hci_send_to_channel(HCI_CHANNEL_MONITOR, skb,
1200249fa169SMarcel Holtmann 						    HCI_SOCK_TRUSTED, NULL);
1201249fa169SMarcel Holtmann 				kfree_skb(skb);
1202249fa169SMarcel Holtmann 			}
1203249fa169SMarcel Holtmann 
1204f9207338SMarcel Holtmann 			hci_sock_set_flag(sk, HCI_MGMT_INDEX_EVENTS);
1205f9207338SMarcel Holtmann 			hci_sock_set_flag(sk, HCI_MGMT_UNCONF_INDEX_EVENTS);
12065504c3a3SMarcel Holtmann 			hci_sock_set_flag(sk, HCI_MGMT_OPTION_EVENTS);
12075504c3a3SMarcel Holtmann 			hci_sock_set_flag(sk, HCI_MGMT_SETTING_EVENTS);
12085504c3a3SMarcel Holtmann 			hci_sock_set_flag(sk, HCI_MGMT_DEV_CLASS_EVENTS);
12095504c3a3SMarcel Holtmann 			hci_sock_set_flag(sk, HCI_MGMT_LOCAL_NAME_EVENTS);
1210f9207338SMarcel Holtmann 		}
1211801c1e8dSJohan Hedberg 		break;
1212801c1e8dSJohan Hedberg 	}
1213801c1e8dSJohan Hedberg 
12141da177e4SLinus Torvalds 	sk->sk_state = BT_BOUND;
12151da177e4SLinus Torvalds 
12161da177e4SLinus Torvalds done:
12171da177e4SLinus Torvalds 	release_sock(sk);
12181da177e4SLinus Torvalds 	return err;
12191da177e4SLinus Torvalds }
12201da177e4SLinus Torvalds 
12218fc9ced3SGustavo Padovan static int hci_sock_getname(struct socket *sock, struct sockaddr *addr,
12228fc9ced3SGustavo Padovan 			    int *addr_len, int peer)
12231da177e4SLinus Torvalds {
12241da177e4SLinus Torvalds 	struct sockaddr_hci *haddr = (struct sockaddr_hci *)addr;
12251da177e4SLinus Torvalds 	struct sock *sk = sock->sk;
12269d4b68b2SMarcel Holtmann 	struct hci_dev *hdev;
12279d4b68b2SMarcel Holtmann 	int err = 0;
12281da177e4SLinus Torvalds 
12291da177e4SLinus Torvalds 	BT_DBG("sock %p sk %p", sock, sk);
12301da177e4SLinus Torvalds 
123106f43cbcSMarcel Holtmann 	if (peer)
123206f43cbcSMarcel Holtmann 		return -EOPNOTSUPP;
123306f43cbcSMarcel Holtmann 
12341da177e4SLinus Torvalds 	lock_sock(sk);
12351da177e4SLinus Torvalds 
12369d4b68b2SMarcel Holtmann 	hdev = hci_pi(sk)->hdev;
12379d4b68b2SMarcel Holtmann 	if (!hdev) {
12389d4b68b2SMarcel Holtmann 		err = -EBADFD;
12399d4b68b2SMarcel Holtmann 		goto done;
12409d4b68b2SMarcel Holtmann 	}
12419d4b68b2SMarcel Holtmann 
12421da177e4SLinus Torvalds 	*addr_len = sizeof(*haddr);
12431da177e4SLinus Torvalds 	haddr->hci_family = AF_BLUETOOTH;
12447b005bd3SMarcel Holtmann 	haddr->hci_dev    = hdev->id;
12459d4b68b2SMarcel Holtmann 	haddr->hci_channel= hci_pi(sk)->channel;
12461da177e4SLinus Torvalds 
12479d4b68b2SMarcel Holtmann done:
12481da177e4SLinus Torvalds 	release_sock(sk);
12499d4b68b2SMarcel Holtmann 	return err;
12501da177e4SLinus Torvalds }
12511da177e4SLinus Torvalds 
12526039aa73SGustavo Padovan static void hci_sock_cmsg(struct sock *sk, struct msghdr *msg,
12536039aa73SGustavo Padovan 			  struct sk_buff *skb)
12541da177e4SLinus Torvalds {
12551da177e4SLinus Torvalds 	__u32 mask = hci_pi(sk)->cmsg_mask;
12561da177e4SLinus Torvalds 
12570d48d939SMarcel Holtmann 	if (mask & HCI_CMSG_DIR) {
12580d48d939SMarcel Holtmann 		int incoming = bt_cb(skb)->incoming;
12598fc9ced3SGustavo Padovan 		put_cmsg(msg, SOL_HCI, HCI_CMSG_DIR, sizeof(incoming),
12608fc9ced3SGustavo Padovan 			 &incoming);
12610d48d939SMarcel Holtmann 	}
12621da177e4SLinus Torvalds 
1263a61bbcf2SPatrick McHardy 	if (mask & HCI_CMSG_TSTAMP) {
1264f6e623a6SJohann Felix Soden #ifdef CONFIG_COMPAT
1265f6e623a6SJohann Felix Soden 		struct compat_timeval ctv;
1266f6e623a6SJohann Felix Soden #endif
1267a61bbcf2SPatrick McHardy 		struct timeval tv;
1268767c5eb5SMarcel Holtmann 		void *data;
1269767c5eb5SMarcel Holtmann 		int len;
1270a61bbcf2SPatrick McHardy 
1271a61bbcf2SPatrick McHardy 		skb_get_timestamp(skb, &tv);
1272767c5eb5SMarcel Holtmann 
12731da97f83SDavid S. Miller 		data = &tv;
12741da97f83SDavid S. Miller 		len = sizeof(tv);
12751da97f83SDavid S. Miller #ifdef CONFIG_COMPAT
1276da88cea1SH. J. Lu 		if (!COMPAT_USE_64BIT_TIME &&
1277da88cea1SH. J. Lu 		    (msg->msg_flags & MSG_CMSG_COMPAT)) {
1278767c5eb5SMarcel Holtmann 			ctv.tv_sec = tv.tv_sec;
1279767c5eb5SMarcel Holtmann 			ctv.tv_usec = tv.tv_usec;
1280767c5eb5SMarcel Holtmann 			data = &ctv;
1281767c5eb5SMarcel Holtmann 			len = sizeof(ctv);
1282767c5eb5SMarcel Holtmann 		}
12831da97f83SDavid S. Miller #endif
1284767c5eb5SMarcel Holtmann 
1285767c5eb5SMarcel Holtmann 		put_cmsg(msg, SOL_HCI, HCI_CMSG_TSTAMP, len, data);
1286a61bbcf2SPatrick McHardy 	}
12871da177e4SLinus Torvalds }
12881da177e4SLinus Torvalds 
12898528d3f7SMarcel Holtmann static int hci_sock_recvmsg(struct socket *sock, struct msghdr *msg,
12908528d3f7SMarcel Holtmann 			    size_t len, int flags)
12911da177e4SLinus Torvalds {
12921da177e4SLinus Torvalds 	int noblock = flags & MSG_DONTWAIT;
12931da177e4SLinus Torvalds 	struct sock *sk = sock->sk;
12941da177e4SLinus Torvalds 	struct sk_buff *skb;
12951da177e4SLinus Torvalds 	int copied, err;
129683871f8cSDenis Kenzior 	unsigned int skblen;
12971da177e4SLinus Torvalds 
12981da177e4SLinus Torvalds 	BT_DBG("sock %p, sk %p", sock, sk);
12991da177e4SLinus Torvalds 
1300d94a6104SMarcel Holtmann 	if (flags & MSG_OOB)
13011da177e4SLinus Torvalds 		return -EOPNOTSUPP;
13021da177e4SLinus Torvalds 
1303ac714949SMarcel Holtmann 	if (hci_pi(sk)->channel == HCI_CHANNEL_LOGGING)
1304ac714949SMarcel Holtmann 		return -EOPNOTSUPP;
1305ac714949SMarcel Holtmann 
13061da177e4SLinus Torvalds 	if (sk->sk_state == BT_CLOSED)
13071da177e4SLinus Torvalds 		return 0;
13081da177e4SLinus Torvalds 
130970f23020SAndrei Emeltchenko 	skb = skb_recv_datagram(sk, flags, noblock, &err);
131070f23020SAndrei Emeltchenko 	if (!skb)
13111da177e4SLinus Torvalds 		return err;
13121da177e4SLinus Torvalds 
131383871f8cSDenis Kenzior 	skblen = skb->len;
13141da177e4SLinus Torvalds 	copied = skb->len;
13151da177e4SLinus Torvalds 	if (len < copied) {
13161da177e4SLinus Torvalds 		msg->msg_flags |= MSG_TRUNC;
13171da177e4SLinus Torvalds 		copied = len;
13181da177e4SLinus Torvalds 	}
13191da177e4SLinus Torvalds 
1320badff6d0SArnaldo Carvalho de Melo 	skb_reset_transport_header(skb);
132151f3d02bSDavid S. Miller 	err = skb_copy_datagram_msg(skb, 0, msg, copied);
13221da177e4SLinus Torvalds 
13233a208627SMarcel Holtmann 	switch (hci_pi(sk)->channel) {
13243a208627SMarcel Holtmann 	case HCI_CHANNEL_RAW:
13251da177e4SLinus Torvalds 		hci_sock_cmsg(sk, msg, skb);
13263a208627SMarcel Holtmann 		break;
132723500189SMarcel Holtmann 	case HCI_CHANNEL_USER:
1328cd82e61cSMarcel Holtmann 	case HCI_CHANNEL_MONITOR:
1329cd82e61cSMarcel Holtmann 		sock_recv_timestamp(msg, sk, skb);
1330cd82e61cSMarcel Holtmann 		break;
1331801c1e8dSJohan Hedberg 	default:
1332801c1e8dSJohan Hedberg 		if (hci_mgmt_chan_find(hci_pi(sk)->channel))
1333801c1e8dSJohan Hedberg 			sock_recv_timestamp(msg, sk, skb);
1334801c1e8dSJohan Hedberg 		break;
13353a208627SMarcel Holtmann 	}
13361da177e4SLinus Torvalds 
13371da177e4SLinus Torvalds 	skb_free_datagram(sk, skb);
13381da177e4SLinus Torvalds 
13394f34228bSLuiz Augusto von Dentz 	if (flags & MSG_TRUNC)
134083871f8cSDenis Kenzior 		copied = skblen;
134183871f8cSDenis Kenzior 
13421da177e4SLinus Torvalds 	return err ? : copied;
13431da177e4SLinus Torvalds }
13441da177e4SLinus Torvalds 
1345fa4335d7SJohan Hedberg static int hci_mgmt_cmd(struct hci_mgmt_chan *chan, struct sock *sk,
1346fa4335d7SJohan Hedberg 			struct msghdr *msg, size_t msglen)
1347fa4335d7SJohan Hedberg {
1348fa4335d7SJohan Hedberg 	void *buf;
1349fa4335d7SJohan Hedberg 	u8 *cp;
1350fa4335d7SJohan Hedberg 	struct mgmt_hdr *hdr;
1351fa4335d7SJohan Hedberg 	u16 opcode, index, len;
1352fa4335d7SJohan Hedberg 	struct hci_dev *hdev = NULL;
1353fa4335d7SJohan Hedberg 	const struct hci_mgmt_handler *handler;
1354fa4335d7SJohan Hedberg 	bool var_len, no_hdev;
1355fa4335d7SJohan Hedberg 	int err;
1356fa4335d7SJohan Hedberg 
1357fa4335d7SJohan Hedberg 	BT_DBG("got %zu bytes", msglen);
1358fa4335d7SJohan Hedberg 
1359fa4335d7SJohan Hedberg 	if (msglen < sizeof(*hdr))
1360fa4335d7SJohan Hedberg 		return -EINVAL;
1361fa4335d7SJohan Hedberg 
1362fa4335d7SJohan Hedberg 	buf = kmalloc(msglen, GFP_KERNEL);
1363fa4335d7SJohan Hedberg 	if (!buf)
1364fa4335d7SJohan Hedberg 		return -ENOMEM;
1365fa4335d7SJohan Hedberg 
1366fa4335d7SJohan Hedberg 	if (memcpy_from_msg(buf, msg, msglen)) {
1367fa4335d7SJohan Hedberg 		err = -EFAULT;
1368fa4335d7SJohan Hedberg 		goto done;
1369fa4335d7SJohan Hedberg 	}
1370fa4335d7SJohan Hedberg 
1371fa4335d7SJohan Hedberg 	hdr = buf;
1372fa4335d7SJohan Hedberg 	opcode = __le16_to_cpu(hdr->opcode);
1373fa4335d7SJohan Hedberg 	index = __le16_to_cpu(hdr->index);
1374fa4335d7SJohan Hedberg 	len = __le16_to_cpu(hdr->len);
1375fa4335d7SJohan Hedberg 
1376fa4335d7SJohan Hedberg 	if (len != msglen - sizeof(*hdr)) {
1377fa4335d7SJohan Hedberg 		err = -EINVAL;
1378fa4335d7SJohan Hedberg 		goto done;
1379fa4335d7SJohan Hedberg 	}
1380fa4335d7SJohan Hedberg 
138138ceaa00SMarcel Holtmann 	if (chan->channel == HCI_CHANNEL_CONTROL) {
138238ceaa00SMarcel Holtmann 		struct sk_buff *skb;
138338ceaa00SMarcel Holtmann 
138438ceaa00SMarcel Holtmann 		/* Send event to monitor */
138538ceaa00SMarcel Holtmann 		skb = create_monitor_ctrl_command(sk, index, opcode, len,
138638ceaa00SMarcel Holtmann 						  buf + sizeof(*hdr));
138738ceaa00SMarcel Holtmann 		if (skb) {
138838ceaa00SMarcel Holtmann 			hci_send_to_channel(HCI_CHANNEL_MONITOR, skb,
138938ceaa00SMarcel Holtmann 					    HCI_SOCK_TRUSTED, NULL);
139038ceaa00SMarcel Holtmann 			kfree_skb(skb);
139138ceaa00SMarcel Holtmann 		}
139238ceaa00SMarcel Holtmann 	}
139338ceaa00SMarcel Holtmann 
1394fa4335d7SJohan Hedberg 	if (opcode >= chan->handler_count ||
1395fa4335d7SJohan Hedberg 	    chan->handlers[opcode].func == NULL) {
1396fa4335d7SJohan Hedberg 		BT_DBG("Unknown op %u", opcode);
1397fa4335d7SJohan Hedberg 		err = mgmt_cmd_status(sk, index, opcode,
1398fa4335d7SJohan Hedberg 				      MGMT_STATUS_UNKNOWN_COMMAND);
1399fa4335d7SJohan Hedberg 		goto done;
1400fa4335d7SJohan Hedberg 	}
1401fa4335d7SJohan Hedberg 
1402fa4335d7SJohan Hedberg 	handler = &chan->handlers[opcode];
1403fa4335d7SJohan Hedberg 
1404fa4335d7SJohan Hedberg 	if (!hci_sock_test_flag(sk, HCI_SOCK_TRUSTED) &&
1405fa4335d7SJohan Hedberg 	    !(handler->flags & HCI_MGMT_UNTRUSTED)) {
1406fa4335d7SJohan Hedberg 		err = mgmt_cmd_status(sk, index, opcode,
1407fa4335d7SJohan Hedberg 				      MGMT_STATUS_PERMISSION_DENIED);
1408fa4335d7SJohan Hedberg 		goto done;
1409fa4335d7SJohan Hedberg 	}
1410fa4335d7SJohan Hedberg 
1411fa4335d7SJohan Hedberg 	if (index != MGMT_INDEX_NONE) {
1412fa4335d7SJohan Hedberg 		hdev = hci_dev_get(index);
1413fa4335d7SJohan Hedberg 		if (!hdev) {
1414fa4335d7SJohan Hedberg 			err = mgmt_cmd_status(sk, index, opcode,
1415fa4335d7SJohan Hedberg 					      MGMT_STATUS_INVALID_INDEX);
1416fa4335d7SJohan Hedberg 			goto done;
1417fa4335d7SJohan Hedberg 		}
1418fa4335d7SJohan Hedberg 
1419fa4335d7SJohan Hedberg 		if (hci_dev_test_flag(hdev, HCI_SETUP) ||
1420fa4335d7SJohan Hedberg 		    hci_dev_test_flag(hdev, HCI_CONFIG) ||
1421fa4335d7SJohan Hedberg 		    hci_dev_test_flag(hdev, HCI_USER_CHANNEL)) {
1422fa4335d7SJohan Hedberg 			err = mgmt_cmd_status(sk, index, opcode,
1423fa4335d7SJohan Hedberg 					      MGMT_STATUS_INVALID_INDEX);
1424fa4335d7SJohan Hedberg 			goto done;
1425fa4335d7SJohan Hedberg 		}
1426fa4335d7SJohan Hedberg 
1427fa4335d7SJohan Hedberg 		if (hci_dev_test_flag(hdev, HCI_UNCONFIGURED) &&
1428fa4335d7SJohan Hedberg 		    !(handler->flags & HCI_MGMT_UNCONFIGURED)) {
1429fa4335d7SJohan Hedberg 			err = mgmt_cmd_status(sk, index, opcode,
1430fa4335d7SJohan Hedberg 					      MGMT_STATUS_INVALID_INDEX);
1431fa4335d7SJohan Hedberg 			goto done;
1432fa4335d7SJohan Hedberg 		}
1433fa4335d7SJohan Hedberg 	}
1434fa4335d7SJohan Hedberg 
1435fa4335d7SJohan Hedberg 	no_hdev = (handler->flags & HCI_MGMT_NO_HDEV);
1436fa4335d7SJohan Hedberg 	if (no_hdev != !hdev) {
1437fa4335d7SJohan Hedberg 		err = mgmt_cmd_status(sk, index, opcode,
1438fa4335d7SJohan Hedberg 				      MGMT_STATUS_INVALID_INDEX);
1439fa4335d7SJohan Hedberg 		goto done;
1440fa4335d7SJohan Hedberg 	}
1441fa4335d7SJohan Hedberg 
1442fa4335d7SJohan Hedberg 	var_len = (handler->flags & HCI_MGMT_VAR_LEN);
1443fa4335d7SJohan Hedberg 	if ((var_len && len < handler->data_len) ||
1444fa4335d7SJohan Hedberg 	    (!var_len && len != handler->data_len)) {
1445fa4335d7SJohan Hedberg 		err = mgmt_cmd_status(sk, index, opcode,
1446fa4335d7SJohan Hedberg 				      MGMT_STATUS_INVALID_PARAMS);
1447fa4335d7SJohan Hedberg 		goto done;
1448fa4335d7SJohan Hedberg 	}
1449fa4335d7SJohan Hedberg 
1450fa4335d7SJohan Hedberg 	if (hdev && chan->hdev_init)
1451fa4335d7SJohan Hedberg 		chan->hdev_init(sk, hdev);
1452fa4335d7SJohan Hedberg 
1453fa4335d7SJohan Hedberg 	cp = buf + sizeof(*hdr);
1454fa4335d7SJohan Hedberg 
1455fa4335d7SJohan Hedberg 	err = handler->func(sk, hdev, cp, len);
1456fa4335d7SJohan Hedberg 	if (err < 0)
1457fa4335d7SJohan Hedberg 		goto done;
1458fa4335d7SJohan Hedberg 
1459fa4335d7SJohan Hedberg 	err = msglen;
1460fa4335d7SJohan Hedberg 
1461fa4335d7SJohan Hedberg done:
1462fa4335d7SJohan Hedberg 	if (hdev)
1463fa4335d7SJohan Hedberg 		hci_dev_put(hdev);
1464fa4335d7SJohan Hedberg 
1465fa4335d7SJohan Hedberg 	kfree(buf);
1466fa4335d7SJohan Hedberg 	return err;
1467fa4335d7SJohan Hedberg }
1468fa4335d7SJohan Hedberg 
1469ac714949SMarcel Holtmann static int hci_logging_frame(struct sock *sk, struct msghdr *msg, int len)
1470ac714949SMarcel Holtmann {
1471ac714949SMarcel Holtmann 	struct hci_mon_hdr *hdr;
1472ac714949SMarcel Holtmann 	struct sk_buff *skb;
1473ac714949SMarcel Holtmann 	struct hci_dev *hdev;
1474ac714949SMarcel Holtmann 	u16 index;
1475ac714949SMarcel Holtmann 	int err;
1476ac714949SMarcel Holtmann 
1477ac714949SMarcel Holtmann 	/* The logging frame consists at minimum of the standard header,
1478ac714949SMarcel Holtmann 	 * the priority byte, the ident length byte and at least one string
1479ac714949SMarcel Holtmann 	 * terminator NUL byte. Anything shorter are invalid packets.
1480ac714949SMarcel Holtmann 	 */
1481ac714949SMarcel Holtmann 	if (len < sizeof(*hdr) + 3)
1482ac714949SMarcel Holtmann 		return -EINVAL;
1483ac714949SMarcel Holtmann 
1484ac714949SMarcel Holtmann 	skb = bt_skb_send_alloc(sk, len, msg->msg_flags & MSG_DONTWAIT, &err);
1485ac714949SMarcel Holtmann 	if (!skb)
1486ac714949SMarcel Holtmann 		return err;
1487ac714949SMarcel Holtmann 
1488ac714949SMarcel Holtmann 	if (memcpy_from_msg(skb_put(skb, len), msg, len)) {
1489ac714949SMarcel Holtmann 		err = -EFAULT;
1490ac714949SMarcel Holtmann 		goto drop;
1491ac714949SMarcel Holtmann 	}
1492ac714949SMarcel Holtmann 
1493ac714949SMarcel Holtmann 	hdr = (void *)skb->data;
1494ac714949SMarcel Holtmann 
1495ac714949SMarcel Holtmann 	if (__le16_to_cpu(hdr->len) != len - sizeof(*hdr)) {
1496ac714949SMarcel Holtmann 		err = -EINVAL;
1497ac714949SMarcel Holtmann 		goto drop;
1498ac714949SMarcel Holtmann 	}
1499ac714949SMarcel Holtmann 
1500ac714949SMarcel Holtmann 	if (__le16_to_cpu(hdr->opcode) == 0x0000) {
1501ac714949SMarcel Holtmann 		__u8 priority = skb->data[sizeof(*hdr)];
1502ac714949SMarcel Holtmann 		__u8 ident_len = skb->data[sizeof(*hdr) + 1];
1503ac714949SMarcel Holtmann 
1504ac714949SMarcel Holtmann 		/* Only the priorities 0-7 are valid and with that any other
1505ac714949SMarcel Holtmann 		 * value results in an invalid packet.
1506ac714949SMarcel Holtmann 		 *
1507ac714949SMarcel Holtmann 		 * The priority byte is followed by an ident length byte and
1508ac714949SMarcel Holtmann 		 * the NUL terminated ident string. Check that the ident
1509ac714949SMarcel Holtmann 		 * length is not overflowing the packet and also that the
1510ac714949SMarcel Holtmann 		 * ident string itself is NUL terminated. In case the ident
1511ac714949SMarcel Holtmann 		 * length is zero, the length value actually doubles as NUL
1512ac714949SMarcel Holtmann 		 * terminator identifier.
1513ac714949SMarcel Holtmann 		 *
1514ac714949SMarcel Holtmann 		 * The message follows the ident string (if present) and
1515ac714949SMarcel Holtmann 		 * must be NUL terminated. Otherwise it is not a valid packet.
1516ac714949SMarcel Holtmann 		 */
1517ac714949SMarcel Holtmann 		if (priority > 7 || skb->data[len - 1] != 0x00 ||
1518ac714949SMarcel Holtmann 		    ident_len > len - sizeof(*hdr) - 3 ||
1519ac714949SMarcel Holtmann 		    skb->data[sizeof(*hdr) + ident_len + 1] != 0x00) {
1520ac714949SMarcel Holtmann 			err = -EINVAL;
1521ac714949SMarcel Holtmann 			goto drop;
1522ac714949SMarcel Holtmann 		}
1523ac714949SMarcel Holtmann 	} else {
1524ac714949SMarcel Holtmann 		err = -EINVAL;
1525ac714949SMarcel Holtmann 		goto drop;
1526ac714949SMarcel Holtmann 	}
1527ac714949SMarcel Holtmann 
1528ac714949SMarcel Holtmann 	index = __le16_to_cpu(hdr->index);
1529ac714949SMarcel Holtmann 
1530ac714949SMarcel Holtmann 	if (index != MGMT_INDEX_NONE) {
1531ac714949SMarcel Holtmann 		hdev = hci_dev_get(index);
1532ac714949SMarcel Holtmann 		if (!hdev) {
1533ac714949SMarcel Holtmann 			err = -ENODEV;
1534ac714949SMarcel Holtmann 			goto drop;
1535ac714949SMarcel Holtmann 		}
1536ac714949SMarcel Holtmann 	} else {
1537ac714949SMarcel Holtmann 		hdev = NULL;
1538ac714949SMarcel Holtmann 	}
1539ac714949SMarcel Holtmann 
1540ac714949SMarcel Holtmann 	hdr->opcode = cpu_to_le16(HCI_MON_USER_LOGGING);
1541ac714949SMarcel Holtmann 
1542ac714949SMarcel Holtmann 	hci_send_to_channel(HCI_CHANNEL_MONITOR, skb, HCI_SOCK_TRUSTED, NULL);
1543ac714949SMarcel Holtmann 	err = len;
1544ac714949SMarcel Holtmann 
1545ac714949SMarcel Holtmann 	if (hdev)
1546ac714949SMarcel Holtmann 		hci_dev_put(hdev);
1547ac714949SMarcel Holtmann 
1548ac714949SMarcel Holtmann drop:
1549ac714949SMarcel Holtmann 	kfree_skb(skb);
1550ac714949SMarcel Holtmann 	return err;
1551ac714949SMarcel Holtmann }
1552ac714949SMarcel Holtmann 
15531b784140SYing Xue static int hci_sock_sendmsg(struct socket *sock, struct msghdr *msg,
15541b784140SYing Xue 			    size_t len)
15551da177e4SLinus Torvalds {
15561da177e4SLinus Torvalds 	struct sock *sk = sock->sk;
1557801c1e8dSJohan Hedberg 	struct hci_mgmt_chan *chan;
15581da177e4SLinus Torvalds 	struct hci_dev *hdev;
15591da177e4SLinus Torvalds 	struct sk_buff *skb;
15601da177e4SLinus Torvalds 	int err;
15611da177e4SLinus Torvalds 
15621da177e4SLinus Torvalds 	BT_DBG("sock %p sk %p", sock, sk);
15631da177e4SLinus Torvalds 
15641da177e4SLinus Torvalds 	if (msg->msg_flags & MSG_OOB)
15651da177e4SLinus Torvalds 		return -EOPNOTSUPP;
15661da177e4SLinus Torvalds 
15671da177e4SLinus Torvalds 	if (msg->msg_flags & ~(MSG_DONTWAIT|MSG_NOSIGNAL|MSG_ERRQUEUE))
15681da177e4SLinus Torvalds 		return -EINVAL;
15691da177e4SLinus Torvalds 
15701da177e4SLinus Torvalds 	if (len < 4 || len > HCI_MAX_FRAME_SIZE)
15711da177e4SLinus Torvalds 		return -EINVAL;
15721da177e4SLinus Torvalds 
15731da177e4SLinus Torvalds 	lock_sock(sk);
15741da177e4SLinus Torvalds 
15750381101fSJohan Hedberg 	switch (hci_pi(sk)->channel) {
15760381101fSJohan Hedberg 	case HCI_CHANNEL_RAW:
157723500189SMarcel Holtmann 	case HCI_CHANNEL_USER:
15780381101fSJohan Hedberg 		break;
1579cd82e61cSMarcel Holtmann 	case HCI_CHANNEL_MONITOR:
1580cd82e61cSMarcel Holtmann 		err = -EOPNOTSUPP;
1581cd82e61cSMarcel Holtmann 		goto done;
1582ac714949SMarcel Holtmann 	case HCI_CHANNEL_LOGGING:
1583ac714949SMarcel Holtmann 		err = hci_logging_frame(sk, msg, len);
1584ac714949SMarcel Holtmann 		goto done;
15850381101fSJohan Hedberg 	default:
1586801c1e8dSJohan Hedberg 		mutex_lock(&mgmt_chan_list_lock);
1587801c1e8dSJohan Hedberg 		chan = __hci_mgmt_chan_find(hci_pi(sk)->channel);
1588801c1e8dSJohan Hedberg 		if (chan)
1589fa4335d7SJohan Hedberg 			err = hci_mgmt_cmd(chan, sk, msg, len);
1590801c1e8dSJohan Hedberg 		else
15910381101fSJohan Hedberg 			err = -EINVAL;
1592801c1e8dSJohan Hedberg 
1593801c1e8dSJohan Hedberg 		mutex_unlock(&mgmt_chan_list_lock);
15940381101fSJohan Hedberg 		goto done;
15950381101fSJohan Hedberg 	}
15960381101fSJohan Hedberg 
159770f23020SAndrei Emeltchenko 	hdev = hci_pi(sk)->hdev;
159870f23020SAndrei Emeltchenko 	if (!hdev) {
15991da177e4SLinus Torvalds 		err = -EBADFD;
16001da177e4SLinus Torvalds 		goto done;
16011da177e4SLinus Torvalds 	}
16021da177e4SLinus Torvalds 
16037e21addcSMarcel Holtmann 	if (!test_bit(HCI_UP, &hdev->flags)) {
16047e21addcSMarcel Holtmann 		err = -ENETDOWN;
16057e21addcSMarcel Holtmann 		goto done;
16067e21addcSMarcel Holtmann 	}
16077e21addcSMarcel Holtmann 
160870f23020SAndrei Emeltchenko 	skb = bt_skb_send_alloc(sk, len, msg->msg_flags & MSG_DONTWAIT, &err);
160970f23020SAndrei Emeltchenko 	if (!skb)
16101da177e4SLinus Torvalds 		goto done;
16111da177e4SLinus Torvalds 
16126ce8e9ceSAl Viro 	if (memcpy_from_msg(skb_put(skb, len), msg, len)) {
16131da177e4SLinus Torvalds 		err = -EFAULT;
16141da177e4SLinus Torvalds 		goto drop;
16151da177e4SLinus Torvalds 	}
16161da177e4SLinus Torvalds 
16178528d3f7SMarcel Holtmann 	hci_skb_pkt_type(skb) = skb->data[0];
16181da177e4SLinus Torvalds 	skb_pull(skb, 1);
16191da177e4SLinus Torvalds 
16201bc5ad16SMarcel Holtmann 	if (hci_pi(sk)->channel == HCI_CHANNEL_USER) {
16211bc5ad16SMarcel Holtmann 		/* No permission check is needed for user channel
16221bc5ad16SMarcel Holtmann 		 * since that gets enforced when binding the socket.
16231bc5ad16SMarcel Holtmann 		 *
16241bc5ad16SMarcel Holtmann 		 * However check that the packet type is valid.
16251bc5ad16SMarcel Holtmann 		 */
1626d79f34e3SMarcel Holtmann 		if (hci_skb_pkt_type(skb) != HCI_COMMAND_PKT &&
1627d79f34e3SMarcel Holtmann 		    hci_skb_pkt_type(skb) != HCI_ACLDATA_PKT &&
1628d79f34e3SMarcel Holtmann 		    hci_skb_pkt_type(skb) != HCI_SCODATA_PKT) {
16291bc5ad16SMarcel Holtmann 			err = -EINVAL;
16301bc5ad16SMarcel Holtmann 			goto drop;
16311bc5ad16SMarcel Holtmann 		}
16321bc5ad16SMarcel Holtmann 
16331bc5ad16SMarcel Holtmann 		skb_queue_tail(&hdev->raw_q, skb);
16341bc5ad16SMarcel Holtmann 		queue_work(hdev->workqueue, &hdev->tx_work);
1635d79f34e3SMarcel Holtmann 	} else if (hci_skb_pkt_type(skb) == HCI_COMMAND_PKT) {
163683985319SHarvey Harrison 		u16 opcode = get_unaligned_le16(skb->data);
16371da177e4SLinus Torvalds 		u16 ogf = hci_opcode_ogf(opcode);
16381da177e4SLinus Torvalds 		u16 ocf = hci_opcode_ocf(opcode);
16391da177e4SLinus Torvalds 
16401da177e4SLinus Torvalds 		if (((ogf > HCI_SFLT_MAX_OGF) ||
16413bb3c755SGustavo Padovan 		     !hci_test_bit(ocf & HCI_FLT_OCF_BITS,
16423bb3c755SGustavo Padovan 				   &hci_sec_filter.ocf_mask[ogf])) &&
16431da177e4SLinus Torvalds 		    !capable(CAP_NET_RAW)) {
16441da177e4SLinus Torvalds 			err = -EPERM;
16451da177e4SLinus Torvalds 			goto drop;
16461da177e4SLinus Torvalds 		}
16471da177e4SLinus Torvalds 
16481982162bSMarcel Holtmann 		/* Since the opcode has already been extracted here, store
16491982162bSMarcel Holtmann 		 * a copy of the value for later use by the drivers.
16501982162bSMarcel Holtmann 		 */
16511982162bSMarcel Holtmann 		hci_skb_opcode(skb) = opcode;
16521982162bSMarcel Holtmann 
1653fee746b0SMarcel Holtmann 		if (ogf == 0x3f) {
16541da177e4SLinus Torvalds 			skb_queue_tail(&hdev->raw_q, skb);
16553eff45eaSGustavo F. Padovan 			queue_work(hdev->workqueue, &hdev->tx_work);
16561da177e4SLinus Torvalds 		} else {
165749c922bbSStephen Hemminger 			/* Stand-alone HCI commands must be flagged as
165811714b3dSJohan Hedberg 			 * single-command requests.
165911714b3dSJohan Hedberg 			 */
166044d27137SJohan Hedberg 			bt_cb(skb)->hci.req_flags |= HCI_REQ_START;
166111714b3dSJohan Hedberg 
16621da177e4SLinus Torvalds 			skb_queue_tail(&hdev->cmd_q, skb);
1663c347b765SGustavo F. Padovan 			queue_work(hdev->workqueue, &hdev->cmd_work);
16641da177e4SLinus Torvalds 		}
16651da177e4SLinus Torvalds 	} else {
16661da177e4SLinus Torvalds 		if (!capable(CAP_NET_RAW)) {
16671da177e4SLinus Torvalds 			err = -EPERM;
16681da177e4SLinus Torvalds 			goto drop;
16691da177e4SLinus Torvalds 		}
16701da177e4SLinus Torvalds 
1671d79f34e3SMarcel Holtmann 		if (hci_skb_pkt_type(skb) != HCI_ACLDATA_PKT &&
1672d79f34e3SMarcel Holtmann 		    hci_skb_pkt_type(skb) != HCI_SCODATA_PKT) {
1673bb77543eSMarcel Holtmann 			err = -EINVAL;
1674bb77543eSMarcel Holtmann 			goto drop;
1675bb77543eSMarcel Holtmann 		}
1676bb77543eSMarcel Holtmann 
16771da177e4SLinus Torvalds 		skb_queue_tail(&hdev->raw_q, skb);
16783eff45eaSGustavo F. Padovan 		queue_work(hdev->workqueue, &hdev->tx_work);
16791da177e4SLinus Torvalds 	}
16801da177e4SLinus Torvalds 
16811da177e4SLinus Torvalds 	err = len;
16821da177e4SLinus Torvalds 
16831da177e4SLinus Torvalds done:
16841da177e4SLinus Torvalds 	release_sock(sk);
16851da177e4SLinus Torvalds 	return err;
16861da177e4SLinus Torvalds 
16871da177e4SLinus Torvalds drop:
16881da177e4SLinus Torvalds 	kfree_skb(skb);
16891da177e4SLinus Torvalds 	goto done;
16901da177e4SLinus Torvalds }
16911da177e4SLinus Torvalds 
16928fc9ced3SGustavo Padovan static int hci_sock_setsockopt(struct socket *sock, int level, int optname,
16938fc9ced3SGustavo Padovan 			       char __user *optval, unsigned int len)
16941da177e4SLinus Torvalds {
16951da177e4SLinus Torvalds 	struct hci_ufilter uf = { .opcode = 0 };
16961da177e4SLinus Torvalds 	struct sock *sk = sock->sk;
16971da177e4SLinus Torvalds 	int err = 0, opt = 0;
16981da177e4SLinus Torvalds 
16991da177e4SLinus Torvalds 	BT_DBG("sk %p, opt %d", sk, optname);
17001da177e4SLinus Torvalds 
170147b0f573SMarcel Holtmann 	if (level != SOL_HCI)
170247b0f573SMarcel Holtmann 		return -ENOPROTOOPT;
170347b0f573SMarcel Holtmann 
17041da177e4SLinus Torvalds 	lock_sock(sk);
17051da177e4SLinus Torvalds 
17062f39cdb7SMarcel Holtmann 	if (hci_pi(sk)->channel != HCI_CHANNEL_RAW) {
1707c2371e80SMarcel Holtmann 		err = -EBADFD;
17082f39cdb7SMarcel Holtmann 		goto done;
17092f39cdb7SMarcel Holtmann 	}
17102f39cdb7SMarcel Holtmann 
17111da177e4SLinus Torvalds 	switch (optname) {
17121da177e4SLinus Torvalds 	case HCI_DATA_DIR:
17131da177e4SLinus Torvalds 		if (get_user(opt, (int __user *)optval)) {
17141da177e4SLinus Torvalds 			err = -EFAULT;
17151da177e4SLinus Torvalds 			break;
17161da177e4SLinus Torvalds 		}
17171da177e4SLinus Torvalds 
17181da177e4SLinus Torvalds 		if (opt)
17191da177e4SLinus Torvalds 			hci_pi(sk)->cmsg_mask |= HCI_CMSG_DIR;
17201da177e4SLinus Torvalds 		else
17211da177e4SLinus Torvalds 			hci_pi(sk)->cmsg_mask &= ~HCI_CMSG_DIR;
17221da177e4SLinus Torvalds 		break;
17231da177e4SLinus Torvalds 
17241da177e4SLinus Torvalds 	case HCI_TIME_STAMP:
17251da177e4SLinus Torvalds 		if (get_user(opt, (int __user *)optval)) {
17261da177e4SLinus Torvalds 			err = -EFAULT;
17271da177e4SLinus Torvalds 			break;
17281da177e4SLinus Torvalds 		}
17291da177e4SLinus Torvalds 
17301da177e4SLinus Torvalds 		if (opt)
17311da177e4SLinus Torvalds 			hci_pi(sk)->cmsg_mask |= HCI_CMSG_TSTAMP;
17321da177e4SLinus Torvalds 		else
17331da177e4SLinus Torvalds 			hci_pi(sk)->cmsg_mask &= ~HCI_CMSG_TSTAMP;
17341da177e4SLinus Torvalds 		break;
17351da177e4SLinus Torvalds 
17361da177e4SLinus Torvalds 	case HCI_FILTER:
17370878b666SMarcel Holtmann 		{
17380878b666SMarcel Holtmann 			struct hci_filter *f = &hci_pi(sk)->filter;
17390878b666SMarcel Holtmann 
17400878b666SMarcel Holtmann 			uf.type_mask = f->type_mask;
17410878b666SMarcel Holtmann 			uf.opcode    = f->opcode;
17420878b666SMarcel Holtmann 			uf.event_mask[0] = *((u32 *) f->event_mask + 0);
17430878b666SMarcel Holtmann 			uf.event_mask[1] = *((u32 *) f->event_mask + 1);
17440878b666SMarcel Holtmann 		}
17450878b666SMarcel Holtmann 
17461da177e4SLinus Torvalds 		len = min_t(unsigned int, len, sizeof(uf));
17471da177e4SLinus Torvalds 		if (copy_from_user(&uf, optval, len)) {
17481da177e4SLinus Torvalds 			err = -EFAULT;
17491da177e4SLinus Torvalds 			break;
17501da177e4SLinus Torvalds 		}
17511da177e4SLinus Torvalds 
17521da177e4SLinus Torvalds 		if (!capable(CAP_NET_RAW)) {
17531da177e4SLinus Torvalds 			uf.type_mask &= hci_sec_filter.type_mask;
17541da177e4SLinus Torvalds 			uf.event_mask[0] &= *((u32 *) hci_sec_filter.event_mask + 0);
17551da177e4SLinus Torvalds 			uf.event_mask[1] &= *((u32 *) hci_sec_filter.event_mask + 1);
17561da177e4SLinus Torvalds 		}
17571da177e4SLinus Torvalds 
17581da177e4SLinus Torvalds 		{
17591da177e4SLinus Torvalds 			struct hci_filter *f = &hci_pi(sk)->filter;
17601da177e4SLinus Torvalds 
17611da177e4SLinus Torvalds 			f->type_mask = uf.type_mask;
17621da177e4SLinus Torvalds 			f->opcode    = uf.opcode;
17631da177e4SLinus Torvalds 			*((u32 *) f->event_mask + 0) = uf.event_mask[0];
17641da177e4SLinus Torvalds 			*((u32 *) f->event_mask + 1) = uf.event_mask[1];
17651da177e4SLinus Torvalds 		}
17661da177e4SLinus Torvalds 		break;
17671da177e4SLinus Torvalds 
17681da177e4SLinus Torvalds 	default:
17691da177e4SLinus Torvalds 		err = -ENOPROTOOPT;
17701da177e4SLinus Torvalds 		break;
17711da177e4SLinus Torvalds 	}
17721da177e4SLinus Torvalds 
17732f39cdb7SMarcel Holtmann done:
17741da177e4SLinus Torvalds 	release_sock(sk);
17751da177e4SLinus Torvalds 	return err;
17761da177e4SLinus Torvalds }
17771da177e4SLinus Torvalds 
17788fc9ced3SGustavo Padovan static int hci_sock_getsockopt(struct socket *sock, int level, int optname,
17798fc9ced3SGustavo Padovan 			       char __user *optval, int __user *optlen)
17801da177e4SLinus Torvalds {
17811da177e4SLinus Torvalds 	struct hci_ufilter uf;
17821da177e4SLinus Torvalds 	struct sock *sk = sock->sk;
1783cedc5469SMarcel Holtmann 	int len, opt, err = 0;
1784cedc5469SMarcel Holtmann 
1785cedc5469SMarcel Holtmann 	BT_DBG("sk %p, opt %d", sk, optname);
17861da177e4SLinus Torvalds 
178747b0f573SMarcel Holtmann 	if (level != SOL_HCI)
178847b0f573SMarcel Holtmann 		return -ENOPROTOOPT;
178947b0f573SMarcel Holtmann 
17901da177e4SLinus Torvalds 	if (get_user(len, optlen))
17911da177e4SLinus Torvalds 		return -EFAULT;
17921da177e4SLinus Torvalds 
1793cedc5469SMarcel Holtmann 	lock_sock(sk);
1794cedc5469SMarcel Holtmann 
1795cedc5469SMarcel Holtmann 	if (hci_pi(sk)->channel != HCI_CHANNEL_RAW) {
1796c2371e80SMarcel Holtmann 		err = -EBADFD;
1797cedc5469SMarcel Holtmann 		goto done;
1798cedc5469SMarcel Holtmann 	}
1799cedc5469SMarcel Holtmann 
18001da177e4SLinus Torvalds 	switch (optname) {
18011da177e4SLinus Torvalds 	case HCI_DATA_DIR:
18021da177e4SLinus Torvalds 		if (hci_pi(sk)->cmsg_mask & HCI_CMSG_DIR)
18031da177e4SLinus Torvalds 			opt = 1;
18041da177e4SLinus Torvalds 		else
18051da177e4SLinus Torvalds 			opt = 0;
18061da177e4SLinus Torvalds 
18071da177e4SLinus Torvalds 		if (put_user(opt, optval))
1808cedc5469SMarcel Holtmann 			err = -EFAULT;
18091da177e4SLinus Torvalds 		break;
18101da177e4SLinus Torvalds 
18111da177e4SLinus Torvalds 	case HCI_TIME_STAMP:
18121da177e4SLinus Torvalds 		if (hci_pi(sk)->cmsg_mask & HCI_CMSG_TSTAMP)
18131da177e4SLinus Torvalds 			opt = 1;
18141da177e4SLinus Torvalds 		else
18151da177e4SLinus Torvalds 			opt = 0;
18161da177e4SLinus Torvalds 
18171da177e4SLinus Torvalds 		if (put_user(opt, optval))
1818cedc5469SMarcel Holtmann 			err = -EFAULT;
18191da177e4SLinus Torvalds 		break;
18201da177e4SLinus Torvalds 
18211da177e4SLinus Torvalds 	case HCI_FILTER:
18221da177e4SLinus Torvalds 		{
18231da177e4SLinus Torvalds 			struct hci_filter *f = &hci_pi(sk)->filter;
18241da177e4SLinus Torvalds 
1825e15ca9a0SMathias Krause 			memset(&uf, 0, sizeof(uf));
18261da177e4SLinus Torvalds 			uf.type_mask = f->type_mask;
18271da177e4SLinus Torvalds 			uf.opcode    = f->opcode;
18281da177e4SLinus Torvalds 			uf.event_mask[0] = *((u32 *) f->event_mask + 0);
18291da177e4SLinus Torvalds 			uf.event_mask[1] = *((u32 *) f->event_mask + 1);
18301da177e4SLinus Torvalds 		}
18311da177e4SLinus Torvalds 
18321da177e4SLinus Torvalds 		len = min_t(unsigned int, len, sizeof(uf));
18331da177e4SLinus Torvalds 		if (copy_to_user(optval, &uf, len))
1834cedc5469SMarcel Holtmann 			err = -EFAULT;
18351da177e4SLinus Torvalds 		break;
18361da177e4SLinus Torvalds 
18371da177e4SLinus Torvalds 	default:
1838cedc5469SMarcel Holtmann 		err = -ENOPROTOOPT;
18391da177e4SLinus Torvalds 		break;
18401da177e4SLinus Torvalds 	}
18411da177e4SLinus Torvalds 
1842cedc5469SMarcel Holtmann done:
1843cedc5469SMarcel Holtmann 	release_sock(sk);
1844cedc5469SMarcel Holtmann 	return err;
18451da177e4SLinus Torvalds }
18461da177e4SLinus Torvalds 
184790ddc4f0SEric Dumazet static const struct proto_ops hci_sock_ops = {
18481da177e4SLinus Torvalds 	.family		= PF_BLUETOOTH,
18491da177e4SLinus Torvalds 	.owner		= THIS_MODULE,
18501da177e4SLinus Torvalds 	.release	= hci_sock_release,
18511da177e4SLinus Torvalds 	.bind		= hci_sock_bind,
18521da177e4SLinus Torvalds 	.getname	= hci_sock_getname,
18531da177e4SLinus Torvalds 	.sendmsg	= hci_sock_sendmsg,
18541da177e4SLinus Torvalds 	.recvmsg	= hci_sock_recvmsg,
18551da177e4SLinus Torvalds 	.ioctl		= hci_sock_ioctl,
18561da177e4SLinus Torvalds 	.poll		= datagram_poll,
18571da177e4SLinus Torvalds 	.listen		= sock_no_listen,
18581da177e4SLinus Torvalds 	.shutdown	= sock_no_shutdown,
18591da177e4SLinus Torvalds 	.setsockopt	= hci_sock_setsockopt,
18601da177e4SLinus Torvalds 	.getsockopt	= hci_sock_getsockopt,
18611da177e4SLinus Torvalds 	.connect	= sock_no_connect,
18621da177e4SLinus Torvalds 	.socketpair	= sock_no_socketpair,
18631da177e4SLinus Torvalds 	.accept		= sock_no_accept,
18641da177e4SLinus Torvalds 	.mmap		= sock_no_mmap
18651da177e4SLinus Torvalds };
18661da177e4SLinus Torvalds 
18671da177e4SLinus Torvalds static struct proto hci_sk_proto = {
18681da177e4SLinus Torvalds 	.name		= "HCI",
18691da177e4SLinus Torvalds 	.owner		= THIS_MODULE,
18701da177e4SLinus Torvalds 	.obj_size	= sizeof(struct hci_pinfo)
18711da177e4SLinus Torvalds };
18721da177e4SLinus Torvalds 
18733f378b68SEric Paris static int hci_sock_create(struct net *net, struct socket *sock, int protocol,
18743f378b68SEric Paris 			   int kern)
18751da177e4SLinus Torvalds {
18761da177e4SLinus Torvalds 	struct sock *sk;
18771da177e4SLinus Torvalds 
18781da177e4SLinus Torvalds 	BT_DBG("sock %p", sock);
18791da177e4SLinus Torvalds 
18801da177e4SLinus Torvalds 	if (sock->type != SOCK_RAW)
18811da177e4SLinus Torvalds 		return -ESOCKTNOSUPPORT;
18821da177e4SLinus Torvalds 
18831da177e4SLinus Torvalds 	sock->ops = &hci_sock_ops;
18841da177e4SLinus Torvalds 
188511aa9c28SEric W. Biederman 	sk = sk_alloc(net, PF_BLUETOOTH, GFP_ATOMIC, &hci_sk_proto, kern);
18861da177e4SLinus Torvalds 	if (!sk)
18871da177e4SLinus Torvalds 		return -ENOMEM;
18881da177e4SLinus Torvalds 
18891da177e4SLinus Torvalds 	sock_init_data(sock, sk);
18901da177e4SLinus Torvalds 
18911da177e4SLinus Torvalds 	sock_reset_flag(sk, SOCK_ZAPPED);
18921da177e4SLinus Torvalds 
18931da177e4SLinus Torvalds 	sk->sk_protocol = protocol;
18941da177e4SLinus Torvalds 
18951da177e4SLinus Torvalds 	sock->state = SS_UNCONNECTED;
18961da177e4SLinus Torvalds 	sk->sk_state = BT_OPEN;
18971da177e4SLinus Torvalds 
18981da177e4SLinus Torvalds 	bt_sock_link(&hci_sk_list, sk);
18991da177e4SLinus Torvalds 	return 0;
19001da177e4SLinus Torvalds }
19011da177e4SLinus Torvalds 
1902ec1b4cf7SStephen Hemminger static const struct net_proto_family hci_sock_family_ops = {
19031da177e4SLinus Torvalds 	.family	= PF_BLUETOOTH,
19041da177e4SLinus Torvalds 	.owner	= THIS_MODULE,
19051da177e4SLinus Torvalds 	.create	= hci_sock_create,
19061da177e4SLinus Torvalds };
19071da177e4SLinus Torvalds 
19081da177e4SLinus Torvalds int __init hci_sock_init(void)
19091da177e4SLinus Torvalds {
19101da177e4SLinus Torvalds 	int err;
19111da177e4SLinus Torvalds 
1912b0a8e282SMarcel Holtmann 	BUILD_BUG_ON(sizeof(struct sockaddr_hci) > sizeof(struct sockaddr));
1913b0a8e282SMarcel Holtmann 
19141da177e4SLinus Torvalds 	err = proto_register(&hci_sk_proto, 0);
19151da177e4SLinus Torvalds 	if (err < 0)
19161da177e4SLinus Torvalds 		return err;
19171da177e4SLinus Torvalds 
19181da177e4SLinus Torvalds 	err = bt_sock_register(BTPROTO_HCI, &hci_sock_family_ops);
1919f7c86637SMasatake YAMATO 	if (err < 0) {
1920f7c86637SMasatake YAMATO 		BT_ERR("HCI socket registration failed");
19211da177e4SLinus Torvalds 		goto error;
1922f7c86637SMasatake YAMATO 	}
1923f7c86637SMasatake YAMATO 
1924b0316615SAl Viro 	err = bt_procfs_init(&init_net, "hci", &hci_sk_list, NULL);
1925f7c86637SMasatake YAMATO 	if (err < 0) {
1926f7c86637SMasatake YAMATO 		BT_ERR("Failed to create HCI proc file");
1927f7c86637SMasatake YAMATO 		bt_sock_unregister(BTPROTO_HCI);
1928f7c86637SMasatake YAMATO 		goto error;
1929f7c86637SMasatake YAMATO 	}
19301da177e4SLinus Torvalds 
19311da177e4SLinus Torvalds 	BT_INFO("HCI socket layer initialized");
19321da177e4SLinus Torvalds 
19331da177e4SLinus Torvalds 	return 0;
19341da177e4SLinus Torvalds 
19351da177e4SLinus Torvalds error:
19361da177e4SLinus Torvalds 	proto_unregister(&hci_sk_proto);
19371da177e4SLinus Torvalds 	return err;
19381da177e4SLinus Torvalds }
19391da177e4SLinus Torvalds 
1940b7440a14SAnand Gadiyar void hci_sock_cleanup(void)
19411da177e4SLinus Torvalds {
1942f7c86637SMasatake YAMATO 	bt_procfs_cleanup(&init_net, "hci");
19435e9d7f86SDavid Herrmann 	bt_sock_unregister(BTPROTO_HCI);
19441da177e4SLinus Torvalds 	proto_unregister(&hci_sk_proto);
19451da177e4SLinus Torvalds }
1946