xref: /openbmc/linux/net/bluetooth/hci_sock.c (revision d0f172b14afa7e0d8a19db5baa07e20b3ec8dcc8)
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>
281da177e4SLinus Torvalds #include <asm/unaligned.h>
291da177e4SLinus Torvalds 
301da177e4SLinus Torvalds #include <net/bluetooth/bluetooth.h>
311da177e4SLinus Torvalds #include <net/bluetooth/hci_core.h>
32cd82e61cSMarcel Holtmann #include <net/bluetooth/hci_mon.h>
331da177e4SLinus Torvalds 
34801c1e8dSJohan Hedberg static LIST_HEAD(mgmt_chan_list);
35801c1e8dSJohan Hedberg static DEFINE_MUTEX(mgmt_chan_list_lock);
36801c1e8dSJohan Hedberg 
37cd82e61cSMarcel Holtmann static atomic_t monitor_promisc = ATOMIC_INIT(0);
38cd82e61cSMarcel Holtmann 
391da177e4SLinus Torvalds /* ----- HCI socket interface ----- */
401da177e4SLinus Torvalds 
41863def58SMarcel Holtmann /* Socket info */
42863def58SMarcel Holtmann #define hci_pi(sk) ((struct hci_pinfo *) sk)
43863def58SMarcel Holtmann 
44863def58SMarcel Holtmann struct hci_pinfo {
45863def58SMarcel Holtmann 	struct bt_sock    bt;
46863def58SMarcel Holtmann 	struct hci_dev    *hdev;
47863def58SMarcel Holtmann 	struct hci_filter filter;
48863def58SMarcel Holtmann 	__u32             cmsg_mask;
49863def58SMarcel Holtmann 	unsigned short    channel;
506befc644SMarcel Holtmann 	unsigned long     flags;
51863def58SMarcel Holtmann };
52863def58SMarcel Holtmann 
536befc644SMarcel Holtmann void hci_sock_set_flag(struct sock *sk, int nr)
546befc644SMarcel Holtmann {
556befc644SMarcel Holtmann 	set_bit(nr, &hci_pi(sk)->flags);
566befc644SMarcel Holtmann }
576befc644SMarcel Holtmann 
586befc644SMarcel Holtmann void hci_sock_clear_flag(struct sock *sk, int nr)
596befc644SMarcel Holtmann {
606befc644SMarcel Holtmann 	clear_bit(nr, &hci_pi(sk)->flags);
616befc644SMarcel Holtmann }
626befc644SMarcel Holtmann 
63c85be545SMarcel Holtmann int hci_sock_test_flag(struct sock *sk, int nr)
64c85be545SMarcel Holtmann {
65c85be545SMarcel Holtmann 	return test_bit(nr, &hci_pi(sk)->flags);
66c85be545SMarcel Holtmann }
67c85be545SMarcel Holtmann 
68*d0f172b1SJohan Hedberg unsigned short hci_sock_get_channel(struct sock *sk)
69*d0f172b1SJohan Hedberg {
70*d0f172b1SJohan Hedberg 	return hci_pi(sk)->channel;
71*d0f172b1SJohan Hedberg }
72*d0f172b1SJohan Hedberg 
739391976aSJiri Slaby static inline int hci_test_bit(int nr, const void *addr)
741da177e4SLinus Torvalds {
759391976aSJiri Slaby 	return *((const __u32 *) addr + (nr >> 5)) & ((__u32) 1 << (nr & 31));
761da177e4SLinus Torvalds }
771da177e4SLinus Torvalds 
781da177e4SLinus Torvalds /* Security filter */
793ad254f7SMarcel Holtmann #define HCI_SFLT_MAX_OGF  5
803ad254f7SMarcel Holtmann 
813ad254f7SMarcel Holtmann struct hci_sec_filter {
823ad254f7SMarcel Holtmann 	__u32 type_mask;
833ad254f7SMarcel Holtmann 	__u32 event_mask[2];
843ad254f7SMarcel Holtmann 	__u32 ocf_mask[HCI_SFLT_MAX_OGF + 1][4];
853ad254f7SMarcel Holtmann };
863ad254f7SMarcel Holtmann 
877e67c112SMarcel Holtmann static const struct hci_sec_filter hci_sec_filter = {
881da177e4SLinus Torvalds 	/* Packet types */
891da177e4SLinus Torvalds 	0x10,
901da177e4SLinus Torvalds 	/* Events */
91dd7f5527SMarcel Holtmann 	{ 0x1000d9fe, 0x0000b00c },
921da177e4SLinus Torvalds 	/* Commands */
931da177e4SLinus Torvalds 	{
941da177e4SLinus Torvalds 		{ 0x0 },
951da177e4SLinus Torvalds 		/* OGF_LINK_CTL */
967c631a67SMarcel Holtmann 		{ 0xbe000006, 0x00000001, 0x00000000, 0x00 },
971da177e4SLinus Torvalds 		/* OGF_LINK_POLICY */
987c631a67SMarcel Holtmann 		{ 0x00005200, 0x00000000, 0x00000000, 0x00 },
991da177e4SLinus Torvalds 		/* OGF_HOST_CTL */
1007c631a67SMarcel Holtmann 		{ 0xaab00200, 0x2b402aaa, 0x05220154, 0x00 },
1011da177e4SLinus Torvalds 		/* OGF_INFO_PARAM */
1027c631a67SMarcel Holtmann 		{ 0x000002be, 0x00000000, 0x00000000, 0x00 },
1031da177e4SLinus Torvalds 		/* OGF_STATUS_PARAM */
1047c631a67SMarcel Holtmann 		{ 0x000000ea, 0x00000000, 0x00000000, 0x00 }
1051da177e4SLinus Torvalds 	}
1061da177e4SLinus Torvalds };
1071da177e4SLinus Torvalds 
1081da177e4SLinus Torvalds static struct bt_sock_list hci_sk_list = {
109d5fb2962SRobert P. J. Day 	.lock = __RW_LOCK_UNLOCKED(hci_sk_list.lock)
1101da177e4SLinus Torvalds };
1111da177e4SLinus Torvalds 
112f81fe64fSMarcel Holtmann static bool is_filtered_packet(struct sock *sk, struct sk_buff *skb)
113f81fe64fSMarcel Holtmann {
114f81fe64fSMarcel Holtmann 	struct hci_filter *flt;
115f81fe64fSMarcel Holtmann 	int flt_type, flt_event;
116f81fe64fSMarcel Holtmann 
117f81fe64fSMarcel Holtmann 	/* Apply filter */
118f81fe64fSMarcel Holtmann 	flt = &hci_pi(sk)->filter;
119f81fe64fSMarcel Holtmann 
120f81fe64fSMarcel Holtmann 	if (bt_cb(skb)->pkt_type == HCI_VENDOR_PKT)
121f81fe64fSMarcel Holtmann 		flt_type = 0;
122f81fe64fSMarcel Holtmann 	else
123f81fe64fSMarcel Holtmann 		flt_type = bt_cb(skb)->pkt_type & HCI_FLT_TYPE_BITS;
124f81fe64fSMarcel Holtmann 
125f81fe64fSMarcel Holtmann 	if (!test_bit(flt_type, &flt->type_mask))
126f81fe64fSMarcel Holtmann 		return true;
127f81fe64fSMarcel Holtmann 
128f81fe64fSMarcel Holtmann 	/* Extra filter for event packets only */
129f81fe64fSMarcel Holtmann 	if (bt_cb(skb)->pkt_type != HCI_EVENT_PKT)
130f81fe64fSMarcel Holtmann 		return false;
131f81fe64fSMarcel Holtmann 
132f81fe64fSMarcel Holtmann 	flt_event = (*(__u8 *)skb->data & HCI_FLT_EVENT_BITS);
133f81fe64fSMarcel Holtmann 
134f81fe64fSMarcel Holtmann 	if (!hci_test_bit(flt_event, &flt->event_mask))
135f81fe64fSMarcel Holtmann 		return true;
136f81fe64fSMarcel Holtmann 
137f81fe64fSMarcel Holtmann 	/* Check filter only when opcode is set */
138f81fe64fSMarcel Holtmann 	if (!flt->opcode)
139f81fe64fSMarcel Holtmann 		return false;
140f81fe64fSMarcel Holtmann 
141f81fe64fSMarcel Holtmann 	if (flt_event == HCI_EV_CMD_COMPLETE &&
142f81fe64fSMarcel Holtmann 	    flt->opcode != get_unaligned((__le16 *)(skb->data + 3)))
143f81fe64fSMarcel Holtmann 		return true;
144f81fe64fSMarcel Holtmann 
145f81fe64fSMarcel Holtmann 	if (flt_event == HCI_EV_CMD_STATUS &&
146f81fe64fSMarcel Holtmann 	    flt->opcode != get_unaligned((__le16 *)(skb->data + 4)))
147f81fe64fSMarcel Holtmann 		return true;
148f81fe64fSMarcel Holtmann 
149f81fe64fSMarcel Holtmann 	return false;
150f81fe64fSMarcel Holtmann }
151f81fe64fSMarcel Holtmann 
1521da177e4SLinus Torvalds /* Send frame to RAW socket */
153470fe1b5SMarcel Holtmann void hci_send_to_sock(struct hci_dev *hdev, struct sk_buff *skb)
1541da177e4SLinus Torvalds {
1551da177e4SLinus Torvalds 	struct sock *sk;
156e0edf373SMarcel Holtmann 	struct sk_buff *skb_copy = NULL;
1571da177e4SLinus Torvalds 
1581da177e4SLinus Torvalds 	BT_DBG("hdev %p len %d", hdev, skb->len);
1591da177e4SLinus Torvalds 
1601da177e4SLinus Torvalds 	read_lock(&hci_sk_list.lock);
161470fe1b5SMarcel Holtmann 
162b67bfe0dSSasha Levin 	sk_for_each(sk, &hci_sk_list.head) {
1631da177e4SLinus Torvalds 		struct sk_buff *nskb;
1641da177e4SLinus Torvalds 
1651da177e4SLinus Torvalds 		if (sk->sk_state != BT_BOUND || hci_pi(sk)->hdev != hdev)
1661da177e4SLinus Torvalds 			continue;
1671da177e4SLinus Torvalds 
1681da177e4SLinus Torvalds 		/* Don't send frame to the socket it came from */
1691da177e4SLinus Torvalds 		if (skb->sk == sk)
1701da177e4SLinus Torvalds 			continue;
1711da177e4SLinus Torvalds 
17223500189SMarcel Holtmann 		if (hci_pi(sk)->channel == HCI_CHANNEL_RAW) {
173f81fe64fSMarcel Holtmann 			if (is_filtered_packet(sk, skb))
1741da177e4SLinus Torvalds 				continue;
17523500189SMarcel Holtmann 		} else if (hci_pi(sk)->channel == HCI_CHANNEL_USER) {
17623500189SMarcel Holtmann 			if (!bt_cb(skb)->incoming)
17723500189SMarcel Holtmann 				continue;
17823500189SMarcel Holtmann 			if (bt_cb(skb)->pkt_type != HCI_EVENT_PKT &&
17923500189SMarcel Holtmann 			    bt_cb(skb)->pkt_type != HCI_ACLDATA_PKT &&
18023500189SMarcel Holtmann 			    bt_cb(skb)->pkt_type != HCI_SCODATA_PKT)
18123500189SMarcel Holtmann 				continue;
18223500189SMarcel Holtmann 		} else {
18323500189SMarcel Holtmann 			/* Don't send frame to other channel types */
18423500189SMarcel Holtmann 			continue;
18523500189SMarcel Holtmann 		}
1861da177e4SLinus Torvalds 
187e0edf373SMarcel Holtmann 		if (!skb_copy) {
188e0edf373SMarcel Holtmann 			/* Create a private copy with headroom */
189bad93e9dSOctavian Purdila 			skb_copy = __pskb_copy_fclone(skb, 1, GFP_ATOMIC, true);
190e0edf373SMarcel Holtmann 			if (!skb_copy)
1911da177e4SLinus Torvalds 				continue;
1921da177e4SLinus Torvalds 
1931da177e4SLinus Torvalds 			/* Put type byte before the data */
194e0edf373SMarcel Holtmann 			memcpy(skb_push(skb_copy, 1), &bt_cb(skb)->pkt_type, 1);
195e0edf373SMarcel Holtmann 		}
196e0edf373SMarcel Holtmann 
197e0edf373SMarcel Holtmann 		nskb = skb_clone(skb_copy, GFP_ATOMIC);
198e0edf373SMarcel Holtmann 		if (!nskb)
199e0edf373SMarcel Holtmann 			continue;
2001da177e4SLinus Torvalds 
2011da177e4SLinus Torvalds 		if (sock_queue_rcv_skb(sk, nskb))
2021da177e4SLinus Torvalds 			kfree_skb(nskb);
2031da177e4SLinus Torvalds 	}
204470fe1b5SMarcel Holtmann 
205470fe1b5SMarcel Holtmann 	read_unlock(&hci_sk_list.lock);
206e0edf373SMarcel Holtmann 
207e0edf373SMarcel Holtmann 	kfree_skb(skb_copy);
208470fe1b5SMarcel Holtmann }
209470fe1b5SMarcel Holtmann 
2107129069eSJohan Hedberg /* Send frame to sockets with specific channel */
2117129069eSJohan Hedberg void hci_send_to_channel(unsigned short channel, struct sk_buff *skb,
212c08b1a1dSMarcel Holtmann 			 int flag, struct sock *skip_sk)
213470fe1b5SMarcel Holtmann {
214470fe1b5SMarcel Holtmann 	struct sock *sk;
215470fe1b5SMarcel Holtmann 
2167129069eSJohan Hedberg 	BT_DBG("channel %u len %d", channel, skb->len);
217470fe1b5SMarcel Holtmann 
218470fe1b5SMarcel Holtmann 	read_lock(&hci_sk_list.lock);
219470fe1b5SMarcel Holtmann 
220b67bfe0dSSasha Levin 	sk_for_each(sk, &hci_sk_list.head) {
221470fe1b5SMarcel Holtmann 		struct sk_buff *nskb;
222470fe1b5SMarcel Holtmann 
223c08b1a1dSMarcel Holtmann 		/* Ignore socket without the flag set */
224c85be545SMarcel Holtmann 		if (!hci_sock_test_flag(sk, flag))
225c08b1a1dSMarcel Holtmann 			continue;
226c08b1a1dSMarcel Holtmann 
227470fe1b5SMarcel Holtmann 		/* Skip the original socket */
228470fe1b5SMarcel Holtmann 		if (sk == skip_sk)
229470fe1b5SMarcel Holtmann 			continue;
230470fe1b5SMarcel Holtmann 
231470fe1b5SMarcel Holtmann 		if (sk->sk_state != BT_BOUND)
232470fe1b5SMarcel Holtmann 			continue;
233470fe1b5SMarcel Holtmann 
2347129069eSJohan Hedberg 		if (hci_pi(sk)->channel != channel)
235d7f72f61SMarcel Holtmann 			continue;
236d7f72f61SMarcel Holtmann 
237d7f72f61SMarcel Holtmann 		nskb = skb_clone(skb, GFP_ATOMIC);
238d7f72f61SMarcel Holtmann 		if (!nskb)
239d7f72f61SMarcel Holtmann 			continue;
240d7f72f61SMarcel Holtmann 
241d7f72f61SMarcel Holtmann 		if (sock_queue_rcv_skb(sk, nskb))
242d7f72f61SMarcel Holtmann 			kfree_skb(nskb);
243d7f72f61SMarcel Holtmann 	}
244d7f72f61SMarcel Holtmann 
245d7f72f61SMarcel Holtmann 	read_unlock(&hci_sk_list.lock);
246d7f72f61SMarcel Holtmann }
247d7f72f61SMarcel Holtmann 
248cd82e61cSMarcel Holtmann /* Send frame to monitor socket */
249cd82e61cSMarcel Holtmann void hci_send_to_monitor(struct hci_dev *hdev, struct sk_buff *skb)
250cd82e61cSMarcel Holtmann {
251cd82e61cSMarcel Holtmann 	struct sk_buff *skb_copy = NULL;
2522b531294SMarcel Holtmann 	struct hci_mon_hdr *hdr;
253cd82e61cSMarcel Holtmann 	__le16 opcode;
254cd82e61cSMarcel Holtmann 
255cd82e61cSMarcel Holtmann 	if (!atomic_read(&monitor_promisc))
256cd82e61cSMarcel Holtmann 		return;
257cd82e61cSMarcel Holtmann 
258cd82e61cSMarcel Holtmann 	BT_DBG("hdev %p len %d", hdev, skb->len);
259cd82e61cSMarcel Holtmann 
260cd82e61cSMarcel Holtmann 	switch (bt_cb(skb)->pkt_type) {
261cd82e61cSMarcel Holtmann 	case HCI_COMMAND_PKT:
262dcf4adbfSJoe Perches 		opcode = cpu_to_le16(HCI_MON_COMMAND_PKT);
263cd82e61cSMarcel Holtmann 		break;
264cd82e61cSMarcel Holtmann 	case HCI_EVENT_PKT:
265dcf4adbfSJoe Perches 		opcode = cpu_to_le16(HCI_MON_EVENT_PKT);
266cd82e61cSMarcel Holtmann 		break;
267cd82e61cSMarcel Holtmann 	case HCI_ACLDATA_PKT:
268cd82e61cSMarcel Holtmann 		if (bt_cb(skb)->incoming)
269dcf4adbfSJoe Perches 			opcode = cpu_to_le16(HCI_MON_ACL_RX_PKT);
270cd82e61cSMarcel Holtmann 		else
271dcf4adbfSJoe Perches 			opcode = cpu_to_le16(HCI_MON_ACL_TX_PKT);
272cd82e61cSMarcel Holtmann 		break;
273cd82e61cSMarcel Holtmann 	case HCI_SCODATA_PKT:
274cd82e61cSMarcel Holtmann 		if (bt_cb(skb)->incoming)
275dcf4adbfSJoe Perches 			opcode = cpu_to_le16(HCI_MON_SCO_RX_PKT);
276cd82e61cSMarcel Holtmann 		else
277dcf4adbfSJoe Perches 			opcode = cpu_to_le16(HCI_MON_SCO_TX_PKT);
278cd82e61cSMarcel Holtmann 		break;
279cd82e61cSMarcel Holtmann 	default:
280cd82e61cSMarcel Holtmann 		return;
281cd82e61cSMarcel Holtmann 	}
282cd82e61cSMarcel Holtmann 
2832b531294SMarcel Holtmann 	/* Create a private copy with headroom */
2842b531294SMarcel Holtmann 	skb_copy = __pskb_copy_fclone(skb, HCI_MON_HDR_SIZE, GFP_ATOMIC, true);
2852b531294SMarcel Holtmann 	if (!skb_copy)
2862b531294SMarcel Holtmann 		return;
2872b531294SMarcel Holtmann 
2882b531294SMarcel Holtmann 	/* Put header before the data */
2892b531294SMarcel Holtmann 	hdr = (void *) skb_push(skb_copy, HCI_MON_HDR_SIZE);
2902b531294SMarcel Holtmann 	hdr->opcode = opcode;
2912b531294SMarcel Holtmann 	hdr->index = cpu_to_le16(hdev->id);
2922b531294SMarcel Holtmann 	hdr->len = cpu_to_le16(skb->len);
2932b531294SMarcel Holtmann 
294c08b1a1dSMarcel Holtmann 	hci_send_to_channel(HCI_CHANNEL_MONITOR, skb_copy,
295c08b1a1dSMarcel Holtmann 			    HCI_SOCK_TRUSTED, NULL);
296cd82e61cSMarcel Holtmann 	kfree_skb(skb_copy);
297cd82e61cSMarcel Holtmann }
298cd82e61cSMarcel Holtmann 
299cd82e61cSMarcel Holtmann static struct sk_buff *create_monitor_event(struct hci_dev *hdev, int event)
300cd82e61cSMarcel Holtmann {
301cd82e61cSMarcel Holtmann 	struct hci_mon_hdr *hdr;
302cd82e61cSMarcel Holtmann 	struct hci_mon_new_index *ni;
303cd82e61cSMarcel Holtmann 	struct sk_buff *skb;
304cd82e61cSMarcel Holtmann 	__le16 opcode;
305cd82e61cSMarcel Holtmann 
306cd82e61cSMarcel Holtmann 	switch (event) {
307cd82e61cSMarcel Holtmann 	case HCI_DEV_REG:
308cd82e61cSMarcel Holtmann 		skb = bt_skb_alloc(HCI_MON_NEW_INDEX_SIZE, GFP_ATOMIC);
309cd82e61cSMarcel Holtmann 		if (!skb)
310cd82e61cSMarcel Holtmann 			return NULL;
311cd82e61cSMarcel Holtmann 
312cd82e61cSMarcel Holtmann 		ni = (void *) skb_put(skb, HCI_MON_NEW_INDEX_SIZE);
313cd82e61cSMarcel Holtmann 		ni->type = hdev->dev_type;
314cd82e61cSMarcel Holtmann 		ni->bus = hdev->bus;
315cd82e61cSMarcel Holtmann 		bacpy(&ni->bdaddr, &hdev->bdaddr);
316cd82e61cSMarcel Holtmann 		memcpy(ni->name, hdev->name, 8);
317cd82e61cSMarcel Holtmann 
318dcf4adbfSJoe Perches 		opcode = cpu_to_le16(HCI_MON_NEW_INDEX);
319cd82e61cSMarcel Holtmann 		break;
320cd82e61cSMarcel Holtmann 
321cd82e61cSMarcel Holtmann 	case HCI_DEV_UNREG:
322cd82e61cSMarcel Holtmann 		skb = bt_skb_alloc(0, GFP_ATOMIC);
323cd82e61cSMarcel Holtmann 		if (!skb)
324cd82e61cSMarcel Holtmann 			return NULL;
325cd82e61cSMarcel Holtmann 
326dcf4adbfSJoe Perches 		opcode = cpu_to_le16(HCI_MON_DEL_INDEX);
327cd82e61cSMarcel Holtmann 		break;
328cd82e61cSMarcel Holtmann 
329cd82e61cSMarcel Holtmann 	default:
330cd82e61cSMarcel Holtmann 		return NULL;
331cd82e61cSMarcel Holtmann 	}
332cd82e61cSMarcel Holtmann 
333cd82e61cSMarcel Holtmann 	__net_timestamp(skb);
334cd82e61cSMarcel Holtmann 
335cd82e61cSMarcel Holtmann 	hdr = (void *) skb_push(skb, HCI_MON_HDR_SIZE);
336cd82e61cSMarcel Holtmann 	hdr->opcode = opcode;
337cd82e61cSMarcel Holtmann 	hdr->index = cpu_to_le16(hdev->id);
338cd82e61cSMarcel Holtmann 	hdr->len = cpu_to_le16(skb->len - HCI_MON_HDR_SIZE);
339cd82e61cSMarcel Holtmann 
340cd82e61cSMarcel Holtmann 	return skb;
341cd82e61cSMarcel Holtmann }
342cd82e61cSMarcel Holtmann 
343cd82e61cSMarcel Holtmann static void send_monitor_replay(struct sock *sk)
344cd82e61cSMarcel Holtmann {
345cd82e61cSMarcel Holtmann 	struct hci_dev *hdev;
346cd82e61cSMarcel Holtmann 
347cd82e61cSMarcel Holtmann 	read_lock(&hci_dev_list_lock);
348cd82e61cSMarcel Holtmann 
349cd82e61cSMarcel Holtmann 	list_for_each_entry(hdev, &hci_dev_list, list) {
350cd82e61cSMarcel Holtmann 		struct sk_buff *skb;
351cd82e61cSMarcel Holtmann 
352cd82e61cSMarcel Holtmann 		skb = create_monitor_event(hdev, HCI_DEV_REG);
353cd82e61cSMarcel Holtmann 		if (!skb)
354cd82e61cSMarcel Holtmann 			continue;
355cd82e61cSMarcel Holtmann 
356cd82e61cSMarcel Holtmann 		if (sock_queue_rcv_skb(sk, skb))
357cd82e61cSMarcel Holtmann 			kfree_skb(skb);
358cd82e61cSMarcel Holtmann 	}
359cd82e61cSMarcel Holtmann 
360cd82e61cSMarcel Holtmann 	read_unlock(&hci_dev_list_lock);
361cd82e61cSMarcel Holtmann }
362cd82e61cSMarcel Holtmann 
363040030efSMarcel Holtmann /* Generate internal stack event */
364040030efSMarcel Holtmann static void hci_si_event(struct hci_dev *hdev, int type, int dlen, void *data)
365040030efSMarcel Holtmann {
366040030efSMarcel Holtmann 	struct hci_event_hdr *hdr;
367040030efSMarcel Holtmann 	struct hci_ev_stack_internal *ev;
368040030efSMarcel Holtmann 	struct sk_buff *skb;
369040030efSMarcel Holtmann 
370040030efSMarcel Holtmann 	skb = bt_skb_alloc(HCI_EVENT_HDR_SIZE + sizeof(*ev) + dlen, GFP_ATOMIC);
371040030efSMarcel Holtmann 	if (!skb)
372040030efSMarcel Holtmann 		return;
373040030efSMarcel Holtmann 
374040030efSMarcel Holtmann 	hdr = (void *) skb_put(skb, HCI_EVENT_HDR_SIZE);
375040030efSMarcel Holtmann 	hdr->evt  = HCI_EV_STACK_INTERNAL;
376040030efSMarcel Holtmann 	hdr->plen = sizeof(*ev) + dlen;
377040030efSMarcel Holtmann 
378040030efSMarcel Holtmann 	ev  = (void *) skb_put(skb, sizeof(*ev) + dlen);
379040030efSMarcel Holtmann 	ev->type = type;
380040030efSMarcel Holtmann 	memcpy(ev->data, data, dlen);
381040030efSMarcel Holtmann 
382040030efSMarcel Holtmann 	bt_cb(skb)->incoming = 1;
383040030efSMarcel Holtmann 	__net_timestamp(skb);
384040030efSMarcel Holtmann 
385040030efSMarcel Holtmann 	bt_cb(skb)->pkt_type = HCI_EVENT_PKT;
386040030efSMarcel Holtmann 	hci_send_to_sock(hdev, skb);
387040030efSMarcel Holtmann 	kfree_skb(skb);
388040030efSMarcel Holtmann }
389040030efSMarcel Holtmann 
390040030efSMarcel Holtmann void hci_sock_dev_event(struct hci_dev *hdev, int event)
391040030efSMarcel Holtmann {
392040030efSMarcel Holtmann 	struct hci_ev_si_device ev;
393040030efSMarcel Holtmann 
394040030efSMarcel Holtmann 	BT_DBG("hdev %s event %d", hdev->name, event);
395040030efSMarcel Holtmann 
396cd82e61cSMarcel Holtmann 	/* Send event to monitor */
397cd82e61cSMarcel Holtmann 	if (atomic_read(&monitor_promisc)) {
398cd82e61cSMarcel Holtmann 		struct sk_buff *skb;
399cd82e61cSMarcel Holtmann 
400cd82e61cSMarcel Holtmann 		skb = create_monitor_event(hdev, event);
401cd82e61cSMarcel Holtmann 		if (skb) {
402c08b1a1dSMarcel Holtmann 			hci_send_to_channel(HCI_CHANNEL_MONITOR, skb,
403c08b1a1dSMarcel Holtmann 					    HCI_SOCK_TRUSTED, NULL);
404cd82e61cSMarcel Holtmann 			kfree_skb(skb);
405cd82e61cSMarcel Holtmann 		}
406cd82e61cSMarcel Holtmann 	}
407cd82e61cSMarcel Holtmann 
408040030efSMarcel Holtmann 	/* Send event to sockets */
409040030efSMarcel Holtmann 	ev.event  = event;
410040030efSMarcel Holtmann 	ev.dev_id = hdev->id;
411040030efSMarcel Holtmann 	hci_si_event(NULL, HCI_EV_SI_DEVICE, sizeof(ev), &ev);
412040030efSMarcel Holtmann 
413040030efSMarcel Holtmann 	if (event == HCI_DEV_UNREG) {
414040030efSMarcel Holtmann 		struct sock *sk;
415040030efSMarcel Holtmann 
416040030efSMarcel Holtmann 		/* Detach sockets from device */
417040030efSMarcel Holtmann 		read_lock(&hci_sk_list.lock);
418b67bfe0dSSasha Levin 		sk_for_each(sk, &hci_sk_list.head) {
419040030efSMarcel Holtmann 			bh_lock_sock_nested(sk);
420040030efSMarcel Holtmann 			if (hci_pi(sk)->hdev == hdev) {
421040030efSMarcel Holtmann 				hci_pi(sk)->hdev = NULL;
422040030efSMarcel Holtmann 				sk->sk_err = EPIPE;
423040030efSMarcel Holtmann 				sk->sk_state = BT_OPEN;
424040030efSMarcel Holtmann 				sk->sk_state_change(sk);
425040030efSMarcel Holtmann 
426040030efSMarcel Holtmann 				hci_dev_put(hdev);
427040030efSMarcel Holtmann 			}
428040030efSMarcel Holtmann 			bh_unlock_sock(sk);
429040030efSMarcel Holtmann 		}
430040030efSMarcel Holtmann 		read_unlock(&hci_sk_list.lock);
431040030efSMarcel Holtmann 	}
432040030efSMarcel Holtmann }
433040030efSMarcel Holtmann 
434801c1e8dSJohan Hedberg static struct hci_mgmt_chan *__hci_mgmt_chan_find(unsigned short channel)
435801c1e8dSJohan Hedberg {
436801c1e8dSJohan Hedberg 	struct hci_mgmt_chan *c;
437801c1e8dSJohan Hedberg 
438801c1e8dSJohan Hedberg 	list_for_each_entry(c, &mgmt_chan_list, list) {
439801c1e8dSJohan Hedberg 		if (c->channel == channel)
440801c1e8dSJohan Hedberg 			return c;
441801c1e8dSJohan Hedberg 	}
442801c1e8dSJohan Hedberg 
443801c1e8dSJohan Hedberg 	return NULL;
444801c1e8dSJohan Hedberg }
445801c1e8dSJohan Hedberg 
446801c1e8dSJohan Hedberg static struct hci_mgmt_chan *hci_mgmt_chan_find(unsigned short channel)
447801c1e8dSJohan Hedberg {
448801c1e8dSJohan Hedberg 	struct hci_mgmt_chan *c;
449801c1e8dSJohan Hedberg 
450801c1e8dSJohan Hedberg 	mutex_lock(&mgmt_chan_list_lock);
451801c1e8dSJohan Hedberg 	c = __hci_mgmt_chan_find(channel);
452801c1e8dSJohan Hedberg 	mutex_unlock(&mgmt_chan_list_lock);
453801c1e8dSJohan Hedberg 
454801c1e8dSJohan Hedberg 	return c;
455801c1e8dSJohan Hedberg }
456801c1e8dSJohan Hedberg 
457801c1e8dSJohan Hedberg int hci_mgmt_chan_register(struct hci_mgmt_chan *c)
458801c1e8dSJohan Hedberg {
459801c1e8dSJohan Hedberg 	if (c->channel < HCI_CHANNEL_CONTROL)
460801c1e8dSJohan Hedberg 		return -EINVAL;
461801c1e8dSJohan Hedberg 
462801c1e8dSJohan Hedberg 	mutex_lock(&mgmt_chan_list_lock);
463801c1e8dSJohan Hedberg 	if (__hci_mgmt_chan_find(c->channel)) {
464801c1e8dSJohan Hedberg 		mutex_unlock(&mgmt_chan_list_lock);
465801c1e8dSJohan Hedberg 		return -EALREADY;
466801c1e8dSJohan Hedberg 	}
467801c1e8dSJohan Hedberg 
468801c1e8dSJohan Hedberg 	list_add_tail(&c->list, &mgmt_chan_list);
469801c1e8dSJohan Hedberg 
470801c1e8dSJohan Hedberg 	mutex_unlock(&mgmt_chan_list_lock);
471801c1e8dSJohan Hedberg 
472801c1e8dSJohan Hedberg 	return 0;
473801c1e8dSJohan Hedberg }
474801c1e8dSJohan Hedberg EXPORT_SYMBOL(hci_mgmt_chan_register);
475801c1e8dSJohan Hedberg 
476801c1e8dSJohan Hedberg void hci_mgmt_chan_unregister(struct hci_mgmt_chan *c)
477801c1e8dSJohan Hedberg {
478801c1e8dSJohan Hedberg 	mutex_lock(&mgmt_chan_list_lock);
479801c1e8dSJohan Hedberg 	list_del(&c->list);
480801c1e8dSJohan Hedberg 	mutex_unlock(&mgmt_chan_list_lock);
481801c1e8dSJohan Hedberg }
482801c1e8dSJohan Hedberg EXPORT_SYMBOL(hci_mgmt_chan_unregister);
483801c1e8dSJohan Hedberg 
4841da177e4SLinus Torvalds static int hci_sock_release(struct socket *sock)
4851da177e4SLinus Torvalds {
4861da177e4SLinus Torvalds 	struct sock *sk = sock->sk;
4877b005bd3SMarcel Holtmann 	struct hci_dev *hdev;
4881da177e4SLinus Torvalds 
4891da177e4SLinus Torvalds 	BT_DBG("sock %p sk %p", sock, sk);
4901da177e4SLinus Torvalds 
4911da177e4SLinus Torvalds 	if (!sk)
4921da177e4SLinus Torvalds 		return 0;
4931da177e4SLinus Torvalds 
4947b005bd3SMarcel Holtmann 	hdev = hci_pi(sk)->hdev;
4957b005bd3SMarcel Holtmann 
496cd82e61cSMarcel Holtmann 	if (hci_pi(sk)->channel == HCI_CHANNEL_MONITOR)
497cd82e61cSMarcel Holtmann 		atomic_dec(&monitor_promisc);
498cd82e61cSMarcel Holtmann 
4991da177e4SLinus Torvalds 	bt_sock_unlink(&hci_sk_list, sk);
5001da177e4SLinus Torvalds 
5011da177e4SLinus Torvalds 	if (hdev) {
50223500189SMarcel Holtmann 		if (hci_pi(sk)->channel == HCI_CHANNEL_USER) {
50323500189SMarcel Holtmann 			mgmt_index_added(hdev);
504a358dc11SMarcel Holtmann 			hci_dev_clear_flag(hdev, HCI_USER_CHANNEL);
50523500189SMarcel Holtmann 			hci_dev_close(hdev->id);
50623500189SMarcel Holtmann 		}
50723500189SMarcel Holtmann 
5081da177e4SLinus Torvalds 		atomic_dec(&hdev->promisc);
5091da177e4SLinus Torvalds 		hci_dev_put(hdev);
5101da177e4SLinus Torvalds 	}
5111da177e4SLinus Torvalds 
5121da177e4SLinus Torvalds 	sock_orphan(sk);
5131da177e4SLinus Torvalds 
5141da177e4SLinus Torvalds 	skb_queue_purge(&sk->sk_receive_queue);
5151da177e4SLinus Torvalds 	skb_queue_purge(&sk->sk_write_queue);
5161da177e4SLinus Torvalds 
5171da177e4SLinus Torvalds 	sock_put(sk);
5181da177e4SLinus Torvalds 	return 0;
5191da177e4SLinus Torvalds }
5201da177e4SLinus Torvalds 
521b2a66aadSAntti Julku static int hci_sock_blacklist_add(struct hci_dev *hdev, void __user *arg)
522f0358568SJohan Hedberg {
523f0358568SJohan Hedberg 	bdaddr_t bdaddr;
5245e762444SAntti Julku 	int err;
525f0358568SJohan Hedberg 
526f0358568SJohan Hedberg 	if (copy_from_user(&bdaddr, arg, sizeof(bdaddr)))
527f0358568SJohan Hedberg 		return -EFAULT;
528f0358568SJohan Hedberg 
52909fd0de5SGustavo F. Padovan 	hci_dev_lock(hdev);
5305e762444SAntti Julku 
531dcc36c16SJohan Hedberg 	err = hci_bdaddr_list_add(&hdev->blacklist, &bdaddr, BDADDR_BREDR);
5325e762444SAntti Julku 
53309fd0de5SGustavo F. Padovan 	hci_dev_unlock(hdev);
5345e762444SAntti Julku 
5355e762444SAntti Julku 	return err;
536f0358568SJohan Hedberg }
537f0358568SJohan Hedberg 
538b2a66aadSAntti Julku static int hci_sock_blacklist_del(struct hci_dev *hdev, void __user *arg)
539f0358568SJohan Hedberg {
540f0358568SJohan Hedberg 	bdaddr_t bdaddr;
5415e762444SAntti Julku 	int err;
542f0358568SJohan Hedberg 
543f0358568SJohan Hedberg 	if (copy_from_user(&bdaddr, arg, sizeof(bdaddr)))
544f0358568SJohan Hedberg 		return -EFAULT;
545f0358568SJohan Hedberg 
54609fd0de5SGustavo F. Padovan 	hci_dev_lock(hdev);
5475e762444SAntti Julku 
548dcc36c16SJohan Hedberg 	err = hci_bdaddr_list_del(&hdev->blacklist, &bdaddr, BDADDR_BREDR);
5495e762444SAntti Julku 
55009fd0de5SGustavo F. Padovan 	hci_dev_unlock(hdev);
5515e762444SAntti Julku 
5525e762444SAntti Julku 	return err;
553f0358568SJohan Hedberg }
554f0358568SJohan Hedberg 
5551da177e4SLinus Torvalds /* Ioctls that require bound socket */
5566039aa73SGustavo Padovan static int hci_sock_bound_ioctl(struct sock *sk, unsigned int cmd,
5576039aa73SGustavo Padovan 				unsigned long arg)
5581da177e4SLinus Torvalds {
5591da177e4SLinus Torvalds 	struct hci_dev *hdev = hci_pi(sk)->hdev;
5601da177e4SLinus Torvalds 
5611da177e4SLinus Torvalds 	if (!hdev)
5621da177e4SLinus Torvalds 		return -EBADFD;
5631da177e4SLinus Torvalds 
564d7a5a11dSMarcel Holtmann 	if (hci_dev_test_flag(hdev, HCI_USER_CHANNEL))
5650736cfa8SMarcel Holtmann 		return -EBUSY;
5660736cfa8SMarcel Holtmann 
567d7a5a11dSMarcel Holtmann 	if (hci_dev_test_flag(hdev, HCI_UNCONFIGURED))
568fee746b0SMarcel Holtmann 		return -EOPNOTSUPP;
569fee746b0SMarcel Holtmann 
5705b69bef5SMarcel Holtmann 	if (hdev->dev_type != HCI_BREDR)
5715b69bef5SMarcel Holtmann 		return -EOPNOTSUPP;
5725b69bef5SMarcel Holtmann 
5731da177e4SLinus Torvalds 	switch (cmd) {
5741da177e4SLinus Torvalds 	case HCISETRAW:
5751da177e4SLinus Torvalds 		if (!capable(CAP_NET_ADMIN))
576bf5b30b8SZhao Hongjiang 			return -EPERM;
577db596681SMarcel Holtmann 		return -EOPNOTSUPP;
5781da177e4SLinus Torvalds 
5791da177e4SLinus Torvalds 	case HCIGETCONNINFO:
5801da177e4SLinus Torvalds 		return hci_get_conn_info(hdev, (void __user *) arg);
5811da177e4SLinus Torvalds 
58240be492fSMarcel Holtmann 	case HCIGETAUTHINFO:
58340be492fSMarcel Holtmann 		return hci_get_auth_info(hdev, (void __user *) arg);
58440be492fSMarcel Holtmann 
585f0358568SJohan Hedberg 	case HCIBLOCKADDR:
586f0358568SJohan Hedberg 		if (!capable(CAP_NET_ADMIN))
587bf5b30b8SZhao Hongjiang 			return -EPERM;
588b2a66aadSAntti Julku 		return hci_sock_blacklist_add(hdev, (void __user *) arg);
589f0358568SJohan Hedberg 
590f0358568SJohan Hedberg 	case HCIUNBLOCKADDR:
591f0358568SJohan Hedberg 		if (!capable(CAP_NET_ADMIN))
592bf5b30b8SZhao Hongjiang 			return -EPERM;
593b2a66aadSAntti Julku 		return hci_sock_blacklist_del(hdev, (void __user *) arg);
5940736cfa8SMarcel Holtmann 	}
595f0358568SJohan Hedberg 
596324d36edSMarcel Holtmann 	return -ENOIOCTLCMD;
5971da177e4SLinus Torvalds }
5981da177e4SLinus Torvalds 
5998fc9ced3SGustavo Padovan static int hci_sock_ioctl(struct socket *sock, unsigned int cmd,
6008fc9ced3SGustavo Padovan 			  unsigned long arg)
6011da177e4SLinus Torvalds {
6021da177e4SLinus Torvalds 	void __user *argp = (void __user *) arg;
6030736cfa8SMarcel Holtmann 	struct sock *sk = sock->sk;
6041da177e4SLinus Torvalds 	int err;
6051da177e4SLinus Torvalds 
6061da177e4SLinus Torvalds 	BT_DBG("cmd %x arg %lx", cmd, arg);
6071da177e4SLinus Torvalds 
608c1c4f956SMarcel Holtmann 	lock_sock(sk);
609c1c4f956SMarcel Holtmann 
610c1c4f956SMarcel Holtmann 	if (hci_pi(sk)->channel != HCI_CHANNEL_RAW) {
611c1c4f956SMarcel Holtmann 		err = -EBADFD;
612c1c4f956SMarcel Holtmann 		goto done;
613c1c4f956SMarcel Holtmann 	}
614c1c4f956SMarcel Holtmann 
615c1c4f956SMarcel Holtmann 	release_sock(sk);
616c1c4f956SMarcel Holtmann 
6171da177e4SLinus Torvalds 	switch (cmd) {
6181da177e4SLinus Torvalds 	case HCIGETDEVLIST:
6191da177e4SLinus Torvalds 		return hci_get_dev_list(argp);
6201da177e4SLinus Torvalds 
6211da177e4SLinus Torvalds 	case HCIGETDEVINFO:
6221da177e4SLinus Torvalds 		return hci_get_dev_info(argp);
6231da177e4SLinus Torvalds 
6241da177e4SLinus Torvalds 	case HCIGETCONNLIST:
6251da177e4SLinus Torvalds 		return hci_get_conn_list(argp);
6261da177e4SLinus Torvalds 
6271da177e4SLinus Torvalds 	case HCIDEVUP:
6281da177e4SLinus Torvalds 		if (!capable(CAP_NET_ADMIN))
629bf5b30b8SZhao Hongjiang 			return -EPERM;
6301da177e4SLinus Torvalds 		return hci_dev_open(arg);
6311da177e4SLinus Torvalds 
6321da177e4SLinus Torvalds 	case HCIDEVDOWN:
6331da177e4SLinus Torvalds 		if (!capable(CAP_NET_ADMIN))
634bf5b30b8SZhao Hongjiang 			return -EPERM;
6351da177e4SLinus Torvalds 		return hci_dev_close(arg);
6361da177e4SLinus Torvalds 
6371da177e4SLinus Torvalds 	case HCIDEVRESET:
6381da177e4SLinus Torvalds 		if (!capable(CAP_NET_ADMIN))
639bf5b30b8SZhao Hongjiang 			return -EPERM;
6401da177e4SLinus Torvalds 		return hci_dev_reset(arg);
6411da177e4SLinus Torvalds 
6421da177e4SLinus Torvalds 	case HCIDEVRESTAT:
6431da177e4SLinus Torvalds 		if (!capable(CAP_NET_ADMIN))
644bf5b30b8SZhao Hongjiang 			return -EPERM;
6451da177e4SLinus Torvalds 		return hci_dev_reset_stat(arg);
6461da177e4SLinus Torvalds 
6471da177e4SLinus Torvalds 	case HCISETSCAN:
6481da177e4SLinus Torvalds 	case HCISETAUTH:
6491da177e4SLinus Torvalds 	case HCISETENCRYPT:
6501da177e4SLinus Torvalds 	case HCISETPTYPE:
6511da177e4SLinus Torvalds 	case HCISETLINKPOL:
6521da177e4SLinus Torvalds 	case HCISETLINKMODE:
6531da177e4SLinus Torvalds 	case HCISETACLMTU:
6541da177e4SLinus Torvalds 	case HCISETSCOMTU:
6551da177e4SLinus Torvalds 		if (!capable(CAP_NET_ADMIN))
656bf5b30b8SZhao Hongjiang 			return -EPERM;
6571da177e4SLinus Torvalds 		return hci_dev_cmd(cmd, argp);
6581da177e4SLinus Torvalds 
6591da177e4SLinus Torvalds 	case HCIINQUIRY:
6601da177e4SLinus Torvalds 		return hci_inquiry(argp);
661c1c4f956SMarcel Holtmann 	}
6621da177e4SLinus Torvalds 
6631da177e4SLinus Torvalds 	lock_sock(sk);
664c1c4f956SMarcel Holtmann 
6651da177e4SLinus Torvalds 	err = hci_sock_bound_ioctl(sk, cmd, arg);
666c1c4f956SMarcel Holtmann 
667c1c4f956SMarcel Holtmann done:
6681da177e4SLinus Torvalds 	release_sock(sk);
6691da177e4SLinus Torvalds 	return err;
6701da177e4SLinus Torvalds }
6711da177e4SLinus Torvalds 
6728fc9ced3SGustavo Padovan static int hci_sock_bind(struct socket *sock, struct sockaddr *addr,
6738fc9ced3SGustavo Padovan 			 int addr_len)
6741da177e4SLinus Torvalds {
6750381101fSJohan Hedberg 	struct sockaddr_hci haddr;
6761da177e4SLinus Torvalds 	struct sock *sk = sock->sk;
6771da177e4SLinus Torvalds 	struct hci_dev *hdev = NULL;
6780381101fSJohan Hedberg 	int len, err = 0;
6791da177e4SLinus Torvalds 
6801da177e4SLinus Torvalds 	BT_DBG("sock %p sk %p", sock, sk);
6811da177e4SLinus Torvalds 
6820381101fSJohan Hedberg 	if (!addr)
6830381101fSJohan Hedberg 		return -EINVAL;
6840381101fSJohan Hedberg 
6850381101fSJohan Hedberg 	memset(&haddr, 0, sizeof(haddr));
6860381101fSJohan Hedberg 	len = min_t(unsigned int, sizeof(haddr), addr_len);
6870381101fSJohan Hedberg 	memcpy(&haddr, addr, len);
6880381101fSJohan Hedberg 
6890381101fSJohan Hedberg 	if (haddr.hci_family != AF_BLUETOOTH)
6900381101fSJohan Hedberg 		return -EINVAL;
6910381101fSJohan Hedberg 
6921da177e4SLinus Torvalds 	lock_sock(sk);
6931da177e4SLinus Torvalds 
6947cc2ade2SMarcel Holtmann 	if (sk->sk_state == BT_BOUND) {
6957cc2ade2SMarcel Holtmann 		err = -EALREADY;
6967cc2ade2SMarcel Holtmann 		goto done;
6977cc2ade2SMarcel Holtmann 	}
6987cc2ade2SMarcel Holtmann 
6997cc2ade2SMarcel Holtmann 	switch (haddr.hci_channel) {
7007cc2ade2SMarcel Holtmann 	case HCI_CHANNEL_RAW:
7017cc2ade2SMarcel Holtmann 		if (hci_pi(sk)->hdev) {
7021da177e4SLinus Torvalds 			err = -EALREADY;
7031da177e4SLinus Torvalds 			goto done;
7041da177e4SLinus Torvalds 		}
7051da177e4SLinus Torvalds 
7060381101fSJohan Hedberg 		if (haddr.hci_dev != HCI_DEV_NONE) {
7070381101fSJohan Hedberg 			hdev = hci_dev_get(haddr.hci_dev);
70870f23020SAndrei Emeltchenko 			if (!hdev) {
7091da177e4SLinus Torvalds 				err = -ENODEV;
7101da177e4SLinus Torvalds 				goto done;
7111da177e4SLinus Torvalds 			}
7121da177e4SLinus Torvalds 
7131da177e4SLinus Torvalds 			atomic_inc(&hdev->promisc);
7141da177e4SLinus Torvalds 		}
7151da177e4SLinus Torvalds 
7161da177e4SLinus Torvalds 		hci_pi(sk)->hdev = hdev;
7177cc2ade2SMarcel Holtmann 		break;
7187cc2ade2SMarcel Holtmann 
71923500189SMarcel Holtmann 	case HCI_CHANNEL_USER:
72023500189SMarcel Holtmann 		if (hci_pi(sk)->hdev) {
72123500189SMarcel Holtmann 			err = -EALREADY;
72223500189SMarcel Holtmann 			goto done;
72323500189SMarcel Holtmann 		}
72423500189SMarcel Holtmann 
72523500189SMarcel Holtmann 		if (haddr.hci_dev == HCI_DEV_NONE) {
72623500189SMarcel Holtmann 			err = -EINVAL;
72723500189SMarcel Holtmann 			goto done;
72823500189SMarcel Holtmann 		}
72923500189SMarcel Holtmann 
73010a8b86fSMarcel Holtmann 		if (!capable(CAP_NET_ADMIN)) {
73123500189SMarcel Holtmann 			err = -EPERM;
73223500189SMarcel Holtmann 			goto done;
73323500189SMarcel Holtmann 		}
73423500189SMarcel Holtmann 
73523500189SMarcel Holtmann 		hdev = hci_dev_get(haddr.hci_dev);
73623500189SMarcel Holtmann 		if (!hdev) {
73723500189SMarcel Holtmann 			err = -ENODEV;
73823500189SMarcel Holtmann 			goto done;
73923500189SMarcel Holtmann 		}
74023500189SMarcel Holtmann 
74123500189SMarcel Holtmann 		if (test_bit(HCI_UP, &hdev->flags) ||
74223500189SMarcel Holtmann 		    test_bit(HCI_INIT, &hdev->flags) ||
743d7a5a11dSMarcel Holtmann 		    hci_dev_test_flag(hdev, HCI_SETUP) ||
744d7a5a11dSMarcel Holtmann 		    hci_dev_test_flag(hdev, HCI_CONFIG)) {
74523500189SMarcel Holtmann 			err = -EBUSY;
74623500189SMarcel Holtmann 			hci_dev_put(hdev);
74723500189SMarcel Holtmann 			goto done;
74823500189SMarcel Holtmann 		}
74923500189SMarcel Holtmann 
750238be788SMarcel Holtmann 		if (hci_dev_test_and_set_flag(hdev, HCI_USER_CHANNEL)) {
75123500189SMarcel Holtmann 			err = -EUSERS;
75223500189SMarcel Holtmann 			hci_dev_put(hdev);
75323500189SMarcel Holtmann 			goto done;
75423500189SMarcel Holtmann 		}
75523500189SMarcel Holtmann 
75623500189SMarcel Holtmann 		mgmt_index_removed(hdev);
75723500189SMarcel Holtmann 
75823500189SMarcel Holtmann 		err = hci_dev_open(hdev->id);
75923500189SMarcel Holtmann 		if (err) {
760a358dc11SMarcel Holtmann 			hci_dev_clear_flag(hdev, HCI_USER_CHANNEL);
761c6521401SMarcel Holtmann 			mgmt_index_added(hdev);
76223500189SMarcel Holtmann 			hci_dev_put(hdev);
76323500189SMarcel Holtmann 			goto done;
76423500189SMarcel Holtmann 		}
76523500189SMarcel Holtmann 
76623500189SMarcel Holtmann 		atomic_inc(&hdev->promisc);
76723500189SMarcel Holtmann 
76823500189SMarcel Holtmann 		hci_pi(sk)->hdev = hdev;
76923500189SMarcel Holtmann 		break;
77023500189SMarcel Holtmann 
771cd82e61cSMarcel Holtmann 	case HCI_CHANNEL_MONITOR:
772cd82e61cSMarcel Holtmann 		if (haddr.hci_dev != HCI_DEV_NONE) {
773cd82e61cSMarcel Holtmann 			err = -EINVAL;
774cd82e61cSMarcel Holtmann 			goto done;
775cd82e61cSMarcel Holtmann 		}
776cd82e61cSMarcel Holtmann 
777cd82e61cSMarcel Holtmann 		if (!capable(CAP_NET_RAW)) {
778cd82e61cSMarcel Holtmann 			err = -EPERM;
779cd82e61cSMarcel Holtmann 			goto done;
780cd82e61cSMarcel Holtmann 		}
781cd82e61cSMarcel Holtmann 
78250ebc055SMarcel Holtmann 		/* The monitor interface is restricted to CAP_NET_RAW
78350ebc055SMarcel Holtmann 		 * capabilities and with that implicitly trusted.
78450ebc055SMarcel Holtmann 		 */
78550ebc055SMarcel Holtmann 		hci_sock_set_flag(sk, HCI_SOCK_TRUSTED);
78650ebc055SMarcel Holtmann 
787cd82e61cSMarcel Holtmann 		send_monitor_replay(sk);
788cd82e61cSMarcel Holtmann 
789cd82e61cSMarcel Holtmann 		atomic_inc(&monitor_promisc);
790cd82e61cSMarcel Holtmann 		break;
791cd82e61cSMarcel Holtmann 
7927cc2ade2SMarcel Holtmann 	default:
793801c1e8dSJohan Hedberg 		if (!hci_mgmt_chan_find(haddr.hci_channel)) {
7947cc2ade2SMarcel Holtmann 			err = -EINVAL;
7957cc2ade2SMarcel Holtmann 			goto done;
7967cc2ade2SMarcel Holtmann 		}
7977cc2ade2SMarcel Holtmann 
798801c1e8dSJohan Hedberg 		if (haddr.hci_dev != HCI_DEV_NONE) {
799801c1e8dSJohan Hedberg 			err = -EINVAL;
800801c1e8dSJohan Hedberg 			goto done;
801801c1e8dSJohan Hedberg 		}
802801c1e8dSJohan Hedberg 
8031195fbb8SMarcel Holtmann 		/* Users with CAP_NET_ADMIN capabilities are allowed
8041195fbb8SMarcel Holtmann 		 * access to all management commands and events. For
8051195fbb8SMarcel Holtmann 		 * untrusted users the interface is restricted and
8061195fbb8SMarcel Holtmann 		 * also only untrusted events are sent.
80750ebc055SMarcel Holtmann 		 */
8081195fbb8SMarcel Holtmann 		if (capable(CAP_NET_ADMIN))
80950ebc055SMarcel Holtmann 			hci_sock_set_flag(sk, HCI_SOCK_TRUSTED);
81050ebc055SMarcel Holtmann 
811f9207338SMarcel Holtmann 		/* At the moment the index and unconfigured index events
812f9207338SMarcel Holtmann 		 * are enabled unconditionally. Setting them on each
813f9207338SMarcel Holtmann 		 * socket when binding keeps this functionality. They
814f9207338SMarcel Holtmann 		 * however might be cleared later and then sending of these
815f9207338SMarcel Holtmann 		 * events will be disabled, but that is then intentional.
816f6b7712eSMarcel Holtmann 		 *
817f6b7712eSMarcel Holtmann 		 * This also enables generic events that are safe to be
818f6b7712eSMarcel Holtmann 		 * received by untrusted users. Example for such events
819f6b7712eSMarcel Holtmann 		 * are changes to settings, class of device, name etc.
820f9207338SMarcel Holtmann 		 */
821f9207338SMarcel Holtmann 		if (haddr.hci_channel == HCI_CHANNEL_CONTROL) {
822f9207338SMarcel Holtmann 			hci_sock_set_flag(sk, HCI_MGMT_INDEX_EVENTS);
823f9207338SMarcel Holtmann 			hci_sock_set_flag(sk, HCI_MGMT_UNCONF_INDEX_EVENTS);
824f6b7712eSMarcel Holtmann 			hci_sock_set_flag(sk, HCI_MGMT_GENERIC_EVENTS);
825f9207338SMarcel Holtmann 		}
826801c1e8dSJohan Hedberg 		break;
827801c1e8dSJohan Hedberg 	}
828801c1e8dSJohan Hedberg 
8297cc2ade2SMarcel Holtmann 
8307cc2ade2SMarcel Holtmann 	hci_pi(sk)->channel = haddr.hci_channel;
8311da177e4SLinus Torvalds 	sk->sk_state = BT_BOUND;
8321da177e4SLinus Torvalds 
8331da177e4SLinus Torvalds done:
8341da177e4SLinus Torvalds 	release_sock(sk);
8351da177e4SLinus Torvalds 	return err;
8361da177e4SLinus Torvalds }
8371da177e4SLinus Torvalds 
8388fc9ced3SGustavo Padovan static int hci_sock_getname(struct socket *sock, struct sockaddr *addr,
8398fc9ced3SGustavo Padovan 			    int *addr_len, int peer)
8401da177e4SLinus Torvalds {
8411da177e4SLinus Torvalds 	struct sockaddr_hci *haddr = (struct sockaddr_hci *) addr;
8421da177e4SLinus Torvalds 	struct sock *sk = sock->sk;
8439d4b68b2SMarcel Holtmann 	struct hci_dev *hdev;
8449d4b68b2SMarcel Holtmann 	int err = 0;
8451da177e4SLinus Torvalds 
8461da177e4SLinus Torvalds 	BT_DBG("sock %p sk %p", sock, sk);
8471da177e4SLinus Torvalds 
84806f43cbcSMarcel Holtmann 	if (peer)
84906f43cbcSMarcel Holtmann 		return -EOPNOTSUPP;
85006f43cbcSMarcel Holtmann 
8511da177e4SLinus Torvalds 	lock_sock(sk);
8521da177e4SLinus Torvalds 
8539d4b68b2SMarcel Holtmann 	hdev = hci_pi(sk)->hdev;
8549d4b68b2SMarcel Holtmann 	if (!hdev) {
8559d4b68b2SMarcel Holtmann 		err = -EBADFD;
8569d4b68b2SMarcel Holtmann 		goto done;
8579d4b68b2SMarcel Holtmann 	}
8589d4b68b2SMarcel Holtmann 
8591da177e4SLinus Torvalds 	*addr_len = sizeof(*haddr);
8601da177e4SLinus Torvalds 	haddr->hci_family = AF_BLUETOOTH;
8617b005bd3SMarcel Holtmann 	haddr->hci_dev    = hdev->id;
8629d4b68b2SMarcel Holtmann 	haddr->hci_channel= hci_pi(sk)->channel;
8631da177e4SLinus Torvalds 
8649d4b68b2SMarcel Holtmann done:
8651da177e4SLinus Torvalds 	release_sock(sk);
8669d4b68b2SMarcel Holtmann 	return err;
8671da177e4SLinus Torvalds }
8681da177e4SLinus Torvalds 
8696039aa73SGustavo Padovan static void hci_sock_cmsg(struct sock *sk, struct msghdr *msg,
8706039aa73SGustavo Padovan 			  struct sk_buff *skb)
8711da177e4SLinus Torvalds {
8721da177e4SLinus Torvalds 	__u32 mask = hci_pi(sk)->cmsg_mask;
8731da177e4SLinus Torvalds 
8740d48d939SMarcel Holtmann 	if (mask & HCI_CMSG_DIR) {
8750d48d939SMarcel Holtmann 		int incoming = bt_cb(skb)->incoming;
8768fc9ced3SGustavo Padovan 		put_cmsg(msg, SOL_HCI, HCI_CMSG_DIR, sizeof(incoming),
8778fc9ced3SGustavo Padovan 			 &incoming);
8780d48d939SMarcel Holtmann 	}
8791da177e4SLinus Torvalds 
880a61bbcf2SPatrick McHardy 	if (mask & HCI_CMSG_TSTAMP) {
881f6e623a6SJohann Felix Soden #ifdef CONFIG_COMPAT
882f6e623a6SJohann Felix Soden 		struct compat_timeval ctv;
883f6e623a6SJohann Felix Soden #endif
884a61bbcf2SPatrick McHardy 		struct timeval tv;
885767c5eb5SMarcel Holtmann 		void *data;
886767c5eb5SMarcel Holtmann 		int len;
887a61bbcf2SPatrick McHardy 
888a61bbcf2SPatrick McHardy 		skb_get_timestamp(skb, &tv);
889767c5eb5SMarcel Holtmann 
8901da97f83SDavid S. Miller 		data = &tv;
8911da97f83SDavid S. Miller 		len = sizeof(tv);
8921da97f83SDavid S. Miller #ifdef CONFIG_COMPAT
893da88cea1SH. J. Lu 		if (!COMPAT_USE_64BIT_TIME &&
894da88cea1SH. J. Lu 		    (msg->msg_flags & MSG_CMSG_COMPAT)) {
895767c5eb5SMarcel Holtmann 			ctv.tv_sec = tv.tv_sec;
896767c5eb5SMarcel Holtmann 			ctv.tv_usec = tv.tv_usec;
897767c5eb5SMarcel Holtmann 			data = &ctv;
898767c5eb5SMarcel Holtmann 			len = sizeof(ctv);
899767c5eb5SMarcel Holtmann 		}
9001da97f83SDavid S. Miller #endif
901767c5eb5SMarcel Holtmann 
902767c5eb5SMarcel Holtmann 		put_cmsg(msg, SOL_HCI, HCI_CMSG_TSTAMP, len, data);
903a61bbcf2SPatrick McHardy 	}
9041da177e4SLinus Torvalds }
9051da177e4SLinus Torvalds 
9061b784140SYing Xue static int hci_sock_recvmsg(struct socket *sock, struct msghdr *msg, size_t len,
9071b784140SYing Xue 			    int flags)
9081da177e4SLinus Torvalds {
9091da177e4SLinus Torvalds 	int noblock = flags & MSG_DONTWAIT;
9101da177e4SLinus Torvalds 	struct sock *sk = sock->sk;
9111da177e4SLinus Torvalds 	struct sk_buff *skb;
9121da177e4SLinus Torvalds 	int copied, err;
9131da177e4SLinus Torvalds 
9141da177e4SLinus Torvalds 	BT_DBG("sock %p, sk %p", sock, sk);
9151da177e4SLinus Torvalds 
9161da177e4SLinus Torvalds 	if (flags & (MSG_OOB))
9171da177e4SLinus Torvalds 		return -EOPNOTSUPP;
9181da177e4SLinus Torvalds 
9191da177e4SLinus Torvalds 	if (sk->sk_state == BT_CLOSED)
9201da177e4SLinus Torvalds 		return 0;
9211da177e4SLinus Torvalds 
92270f23020SAndrei Emeltchenko 	skb = skb_recv_datagram(sk, flags, noblock, &err);
92370f23020SAndrei Emeltchenko 	if (!skb)
9241da177e4SLinus Torvalds 		return err;
9251da177e4SLinus Torvalds 
9261da177e4SLinus Torvalds 	copied = skb->len;
9271da177e4SLinus Torvalds 	if (len < copied) {
9281da177e4SLinus Torvalds 		msg->msg_flags |= MSG_TRUNC;
9291da177e4SLinus Torvalds 		copied = len;
9301da177e4SLinus Torvalds 	}
9311da177e4SLinus Torvalds 
932badff6d0SArnaldo Carvalho de Melo 	skb_reset_transport_header(skb);
93351f3d02bSDavid S. Miller 	err = skb_copy_datagram_msg(skb, 0, msg, copied);
9341da177e4SLinus Torvalds 
9353a208627SMarcel Holtmann 	switch (hci_pi(sk)->channel) {
9363a208627SMarcel Holtmann 	case HCI_CHANNEL_RAW:
9371da177e4SLinus Torvalds 		hci_sock_cmsg(sk, msg, skb);
9383a208627SMarcel Holtmann 		break;
93923500189SMarcel Holtmann 	case HCI_CHANNEL_USER:
940cd82e61cSMarcel Holtmann 	case HCI_CHANNEL_MONITOR:
941cd82e61cSMarcel Holtmann 		sock_recv_timestamp(msg, sk, skb);
942cd82e61cSMarcel Holtmann 		break;
943801c1e8dSJohan Hedberg 	default:
944801c1e8dSJohan Hedberg 		if (hci_mgmt_chan_find(hci_pi(sk)->channel))
945801c1e8dSJohan Hedberg 			sock_recv_timestamp(msg, sk, skb);
946801c1e8dSJohan Hedberg 		break;
9473a208627SMarcel Holtmann 	}
9481da177e4SLinus Torvalds 
9491da177e4SLinus Torvalds 	skb_free_datagram(sk, skb);
9501da177e4SLinus Torvalds 
9511da177e4SLinus Torvalds 	return err ? : copied;
9521da177e4SLinus Torvalds }
9531da177e4SLinus Torvalds 
9541b784140SYing Xue static int hci_sock_sendmsg(struct socket *sock, struct msghdr *msg,
9551b784140SYing Xue 			    size_t len)
9561da177e4SLinus Torvalds {
9571da177e4SLinus Torvalds 	struct sock *sk = sock->sk;
958801c1e8dSJohan Hedberg 	struct hci_mgmt_chan *chan;
9591da177e4SLinus Torvalds 	struct hci_dev *hdev;
9601da177e4SLinus Torvalds 	struct sk_buff *skb;
9611da177e4SLinus Torvalds 	int err;
9621da177e4SLinus Torvalds 
9631da177e4SLinus Torvalds 	BT_DBG("sock %p sk %p", sock, sk);
9641da177e4SLinus Torvalds 
9651da177e4SLinus Torvalds 	if (msg->msg_flags & MSG_OOB)
9661da177e4SLinus Torvalds 		return -EOPNOTSUPP;
9671da177e4SLinus Torvalds 
9681da177e4SLinus Torvalds 	if (msg->msg_flags & ~(MSG_DONTWAIT|MSG_NOSIGNAL|MSG_ERRQUEUE))
9691da177e4SLinus Torvalds 		return -EINVAL;
9701da177e4SLinus Torvalds 
9711da177e4SLinus Torvalds 	if (len < 4 || len > HCI_MAX_FRAME_SIZE)
9721da177e4SLinus Torvalds 		return -EINVAL;
9731da177e4SLinus Torvalds 
9741da177e4SLinus Torvalds 	lock_sock(sk);
9751da177e4SLinus Torvalds 
9760381101fSJohan Hedberg 	switch (hci_pi(sk)->channel) {
9770381101fSJohan Hedberg 	case HCI_CHANNEL_RAW:
97823500189SMarcel Holtmann 	case HCI_CHANNEL_USER:
9790381101fSJohan Hedberg 		break;
980cd82e61cSMarcel Holtmann 	case HCI_CHANNEL_MONITOR:
981cd82e61cSMarcel Holtmann 		err = -EOPNOTSUPP;
982cd82e61cSMarcel Holtmann 		goto done;
9830381101fSJohan Hedberg 	default:
984801c1e8dSJohan Hedberg 		mutex_lock(&mgmt_chan_list_lock);
985801c1e8dSJohan Hedberg 		chan = __hci_mgmt_chan_find(hci_pi(sk)->channel);
986801c1e8dSJohan Hedberg 		if (chan)
9876d785aa3SJohan Hedberg 			err = mgmt_control(chan, sk, msg, len);
988801c1e8dSJohan Hedberg 		else
9890381101fSJohan Hedberg 			err = -EINVAL;
990801c1e8dSJohan Hedberg 
991801c1e8dSJohan Hedberg 		mutex_unlock(&mgmt_chan_list_lock);
9920381101fSJohan Hedberg 		goto done;
9930381101fSJohan Hedberg 	}
9940381101fSJohan Hedberg 
99570f23020SAndrei Emeltchenko 	hdev = hci_pi(sk)->hdev;
99670f23020SAndrei Emeltchenko 	if (!hdev) {
9971da177e4SLinus Torvalds 		err = -EBADFD;
9981da177e4SLinus Torvalds 		goto done;
9991da177e4SLinus Torvalds 	}
10001da177e4SLinus Torvalds 
10017e21addcSMarcel Holtmann 	if (!test_bit(HCI_UP, &hdev->flags)) {
10027e21addcSMarcel Holtmann 		err = -ENETDOWN;
10037e21addcSMarcel Holtmann 		goto done;
10047e21addcSMarcel Holtmann 	}
10057e21addcSMarcel Holtmann 
100670f23020SAndrei Emeltchenko 	skb = bt_skb_send_alloc(sk, len, msg->msg_flags & MSG_DONTWAIT, &err);
100770f23020SAndrei Emeltchenko 	if (!skb)
10081da177e4SLinus Torvalds 		goto done;
10091da177e4SLinus Torvalds 
10106ce8e9ceSAl Viro 	if (memcpy_from_msg(skb_put(skb, len), msg, len)) {
10111da177e4SLinus Torvalds 		err = -EFAULT;
10121da177e4SLinus Torvalds 		goto drop;
10131da177e4SLinus Torvalds 	}
10141da177e4SLinus Torvalds 
10150d48d939SMarcel Holtmann 	bt_cb(skb)->pkt_type = *((unsigned char *) skb->data);
10161da177e4SLinus Torvalds 	skb_pull(skb, 1);
10171da177e4SLinus Torvalds 
10181bc5ad16SMarcel Holtmann 	if (hci_pi(sk)->channel == HCI_CHANNEL_USER) {
10191bc5ad16SMarcel Holtmann 		/* No permission check is needed for user channel
10201bc5ad16SMarcel Holtmann 		 * since that gets enforced when binding the socket.
10211bc5ad16SMarcel Holtmann 		 *
10221bc5ad16SMarcel Holtmann 		 * However check that the packet type is valid.
10231bc5ad16SMarcel Holtmann 		 */
10241bc5ad16SMarcel Holtmann 		if (bt_cb(skb)->pkt_type != HCI_COMMAND_PKT &&
10251bc5ad16SMarcel Holtmann 		    bt_cb(skb)->pkt_type != HCI_ACLDATA_PKT &&
10261bc5ad16SMarcel Holtmann 		    bt_cb(skb)->pkt_type != HCI_SCODATA_PKT) {
10271bc5ad16SMarcel Holtmann 			err = -EINVAL;
10281bc5ad16SMarcel Holtmann 			goto drop;
10291bc5ad16SMarcel Holtmann 		}
10301bc5ad16SMarcel Holtmann 
10311bc5ad16SMarcel Holtmann 		skb_queue_tail(&hdev->raw_q, skb);
10321bc5ad16SMarcel Holtmann 		queue_work(hdev->workqueue, &hdev->tx_work);
10331bc5ad16SMarcel Holtmann 	} else if (bt_cb(skb)->pkt_type == HCI_COMMAND_PKT) {
103483985319SHarvey Harrison 		u16 opcode = get_unaligned_le16(skb->data);
10351da177e4SLinus Torvalds 		u16 ogf = hci_opcode_ogf(opcode);
10361da177e4SLinus Torvalds 		u16 ocf = hci_opcode_ocf(opcode);
10371da177e4SLinus Torvalds 
10381da177e4SLinus Torvalds 		if (((ogf > HCI_SFLT_MAX_OGF) ||
10393bb3c755SGustavo Padovan 		     !hci_test_bit(ocf & HCI_FLT_OCF_BITS,
10403bb3c755SGustavo Padovan 				   &hci_sec_filter.ocf_mask[ogf])) &&
10411da177e4SLinus Torvalds 		    !capable(CAP_NET_RAW)) {
10421da177e4SLinus Torvalds 			err = -EPERM;
10431da177e4SLinus Torvalds 			goto drop;
10441da177e4SLinus Torvalds 		}
10451da177e4SLinus Torvalds 
1046fee746b0SMarcel Holtmann 		if (ogf == 0x3f) {
10471da177e4SLinus Torvalds 			skb_queue_tail(&hdev->raw_q, skb);
10483eff45eaSGustavo F. Padovan 			queue_work(hdev->workqueue, &hdev->tx_work);
10491da177e4SLinus Torvalds 		} else {
105049c922bbSStephen Hemminger 			/* Stand-alone HCI commands must be flagged as
105111714b3dSJohan Hedberg 			 * single-command requests.
105211714b3dSJohan Hedberg 			 */
10536368c235SEyal Birger 			bt_cb(skb)->req_start = 1;
105411714b3dSJohan Hedberg 
10551da177e4SLinus Torvalds 			skb_queue_tail(&hdev->cmd_q, skb);
1056c347b765SGustavo F. Padovan 			queue_work(hdev->workqueue, &hdev->cmd_work);
10571da177e4SLinus Torvalds 		}
10581da177e4SLinus Torvalds 	} else {
10591da177e4SLinus Torvalds 		if (!capable(CAP_NET_RAW)) {
10601da177e4SLinus Torvalds 			err = -EPERM;
10611da177e4SLinus Torvalds 			goto drop;
10621da177e4SLinus Torvalds 		}
10631da177e4SLinus Torvalds 
10641da177e4SLinus Torvalds 		skb_queue_tail(&hdev->raw_q, skb);
10653eff45eaSGustavo F. Padovan 		queue_work(hdev->workqueue, &hdev->tx_work);
10661da177e4SLinus Torvalds 	}
10671da177e4SLinus Torvalds 
10681da177e4SLinus Torvalds 	err = len;
10691da177e4SLinus Torvalds 
10701da177e4SLinus Torvalds done:
10711da177e4SLinus Torvalds 	release_sock(sk);
10721da177e4SLinus Torvalds 	return err;
10731da177e4SLinus Torvalds 
10741da177e4SLinus Torvalds drop:
10751da177e4SLinus Torvalds 	kfree_skb(skb);
10761da177e4SLinus Torvalds 	goto done;
10771da177e4SLinus Torvalds }
10781da177e4SLinus Torvalds 
10798fc9ced3SGustavo Padovan static int hci_sock_setsockopt(struct socket *sock, int level, int optname,
10808fc9ced3SGustavo Padovan 			       char __user *optval, unsigned int len)
10811da177e4SLinus Torvalds {
10821da177e4SLinus Torvalds 	struct hci_ufilter uf = { .opcode = 0 };
10831da177e4SLinus Torvalds 	struct sock *sk = sock->sk;
10841da177e4SLinus Torvalds 	int err = 0, opt = 0;
10851da177e4SLinus Torvalds 
10861da177e4SLinus Torvalds 	BT_DBG("sk %p, opt %d", sk, optname);
10871da177e4SLinus Torvalds 
10881da177e4SLinus Torvalds 	lock_sock(sk);
10891da177e4SLinus Torvalds 
10902f39cdb7SMarcel Holtmann 	if (hci_pi(sk)->channel != HCI_CHANNEL_RAW) {
1091c2371e80SMarcel Holtmann 		err = -EBADFD;
10922f39cdb7SMarcel Holtmann 		goto done;
10932f39cdb7SMarcel Holtmann 	}
10942f39cdb7SMarcel Holtmann 
10951da177e4SLinus Torvalds 	switch (optname) {
10961da177e4SLinus Torvalds 	case HCI_DATA_DIR:
10971da177e4SLinus Torvalds 		if (get_user(opt, (int __user *)optval)) {
10981da177e4SLinus Torvalds 			err = -EFAULT;
10991da177e4SLinus Torvalds 			break;
11001da177e4SLinus Torvalds 		}
11011da177e4SLinus Torvalds 
11021da177e4SLinus Torvalds 		if (opt)
11031da177e4SLinus Torvalds 			hci_pi(sk)->cmsg_mask |= HCI_CMSG_DIR;
11041da177e4SLinus Torvalds 		else
11051da177e4SLinus Torvalds 			hci_pi(sk)->cmsg_mask &= ~HCI_CMSG_DIR;
11061da177e4SLinus Torvalds 		break;
11071da177e4SLinus Torvalds 
11081da177e4SLinus Torvalds 	case HCI_TIME_STAMP:
11091da177e4SLinus Torvalds 		if (get_user(opt, (int __user *)optval)) {
11101da177e4SLinus Torvalds 			err = -EFAULT;
11111da177e4SLinus Torvalds 			break;
11121da177e4SLinus Torvalds 		}
11131da177e4SLinus Torvalds 
11141da177e4SLinus Torvalds 		if (opt)
11151da177e4SLinus Torvalds 			hci_pi(sk)->cmsg_mask |= HCI_CMSG_TSTAMP;
11161da177e4SLinus Torvalds 		else
11171da177e4SLinus Torvalds 			hci_pi(sk)->cmsg_mask &= ~HCI_CMSG_TSTAMP;
11181da177e4SLinus Torvalds 		break;
11191da177e4SLinus Torvalds 
11201da177e4SLinus Torvalds 	case HCI_FILTER:
11210878b666SMarcel Holtmann 		{
11220878b666SMarcel Holtmann 			struct hci_filter *f = &hci_pi(sk)->filter;
11230878b666SMarcel Holtmann 
11240878b666SMarcel Holtmann 			uf.type_mask = f->type_mask;
11250878b666SMarcel Holtmann 			uf.opcode    = f->opcode;
11260878b666SMarcel Holtmann 			uf.event_mask[0] = *((u32 *) f->event_mask + 0);
11270878b666SMarcel Holtmann 			uf.event_mask[1] = *((u32 *) f->event_mask + 1);
11280878b666SMarcel Holtmann 		}
11290878b666SMarcel Holtmann 
11301da177e4SLinus Torvalds 		len = min_t(unsigned int, len, sizeof(uf));
11311da177e4SLinus Torvalds 		if (copy_from_user(&uf, optval, len)) {
11321da177e4SLinus Torvalds 			err = -EFAULT;
11331da177e4SLinus Torvalds 			break;
11341da177e4SLinus Torvalds 		}
11351da177e4SLinus Torvalds 
11361da177e4SLinus Torvalds 		if (!capable(CAP_NET_RAW)) {
11371da177e4SLinus Torvalds 			uf.type_mask &= hci_sec_filter.type_mask;
11381da177e4SLinus Torvalds 			uf.event_mask[0] &= *((u32 *) hci_sec_filter.event_mask + 0);
11391da177e4SLinus Torvalds 			uf.event_mask[1] &= *((u32 *) hci_sec_filter.event_mask + 1);
11401da177e4SLinus Torvalds 		}
11411da177e4SLinus Torvalds 
11421da177e4SLinus Torvalds 		{
11431da177e4SLinus Torvalds 			struct hci_filter *f = &hci_pi(sk)->filter;
11441da177e4SLinus Torvalds 
11451da177e4SLinus Torvalds 			f->type_mask = uf.type_mask;
11461da177e4SLinus Torvalds 			f->opcode    = uf.opcode;
11471da177e4SLinus Torvalds 			*((u32 *) f->event_mask + 0) = uf.event_mask[0];
11481da177e4SLinus Torvalds 			*((u32 *) f->event_mask + 1) = uf.event_mask[1];
11491da177e4SLinus Torvalds 		}
11501da177e4SLinus Torvalds 		break;
11511da177e4SLinus Torvalds 
11521da177e4SLinus Torvalds 	default:
11531da177e4SLinus Torvalds 		err = -ENOPROTOOPT;
11541da177e4SLinus Torvalds 		break;
11551da177e4SLinus Torvalds 	}
11561da177e4SLinus Torvalds 
11572f39cdb7SMarcel Holtmann done:
11581da177e4SLinus Torvalds 	release_sock(sk);
11591da177e4SLinus Torvalds 	return err;
11601da177e4SLinus Torvalds }
11611da177e4SLinus Torvalds 
11628fc9ced3SGustavo Padovan static int hci_sock_getsockopt(struct socket *sock, int level, int optname,
11638fc9ced3SGustavo Padovan 			       char __user *optval, int __user *optlen)
11641da177e4SLinus Torvalds {
11651da177e4SLinus Torvalds 	struct hci_ufilter uf;
11661da177e4SLinus Torvalds 	struct sock *sk = sock->sk;
1167cedc5469SMarcel Holtmann 	int len, opt, err = 0;
1168cedc5469SMarcel Holtmann 
1169cedc5469SMarcel Holtmann 	BT_DBG("sk %p, opt %d", sk, optname);
11701da177e4SLinus Torvalds 
11711da177e4SLinus Torvalds 	if (get_user(len, optlen))
11721da177e4SLinus Torvalds 		return -EFAULT;
11731da177e4SLinus Torvalds 
1174cedc5469SMarcel Holtmann 	lock_sock(sk);
1175cedc5469SMarcel Holtmann 
1176cedc5469SMarcel Holtmann 	if (hci_pi(sk)->channel != HCI_CHANNEL_RAW) {
1177c2371e80SMarcel Holtmann 		err = -EBADFD;
1178cedc5469SMarcel Holtmann 		goto done;
1179cedc5469SMarcel Holtmann 	}
1180cedc5469SMarcel Holtmann 
11811da177e4SLinus Torvalds 	switch (optname) {
11821da177e4SLinus Torvalds 	case HCI_DATA_DIR:
11831da177e4SLinus Torvalds 		if (hci_pi(sk)->cmsg_mask & HCI_CMSG_DIR)
11841da177e4SLinus Torvalds 			opt = 1;
11851da177e4SLinus Torvalds 		else
11861da177e4SLinus Torvalds 			opt = 0;
11871da177e4SLinus Torvalds 
11881da177e4SLinus Torvalds 		if (put_user(opt, optval))
1189cedc5469SMarcel Holtmann 			err = -EFAULT;
11901da177e4SLinus Torvalds 		break;
11911da177e4SLinus Torvalds 
11921da177e4SLinus Torvalds 	case HCI_TIME_STAMP:
11931da177e4SLinus Torvalds 		if (hci_pi(sk)->cmsg_mask & HCI_CMSG_TSTAMP)
11941da177e4SLinus Torvalds 			opt = 1;
11951da177e4SLinus Torvalds 		else
11961da177e4SLinus Torvalds 			opt = 0;
11971da177e4SLinus Torvalds 
11981da177e4SLinus Torvalds 		if (put_user(opt, optval))
1199cedc5469SMarcel Holtmann 			err = -EFAULT;
12001da177e4SLinus Torvalds 		break;
12011da177e4SLinus Torvalds 
12021da177e4SLinus Torvalds 	case HCI_FILTER:
12031da177e4SLinus Torvalds 		{
12041da177e4SLinus Torvalds 			struct hci_filter *f = &hci_pi(sk)->filter;
12051da177e4SLinus Torvalds 
1206e15ca9a0SMathias Krause 			memset(&uf, 0, sizeof(uf));
12071da177e4SLinus Torvalds 			uf.type_mask = f->type_mask;
12081da177e4SLinus Torvalds 			uf.opcode    = f->opcode;
12091da177e4SLinus Torvalds 			uf.event_mask[0] = *((u32 *) f->event_mask + 0);
12101da177e4SLinus Torvalds 			uf.event_mask[1] = *((u32 *) f->event_mask + 1);
12111da177e4SLinus Torvalds 		}
12121da177e4SLinus Torvalds 
12131da177e4SLinus Torvalds 		len = min_t(unsigned int, len, sizeof(uf));
12141da177e4SLinus Torvalds 		if (copy_to_user(optval, &uf, len))
1215cedc5469SMarcel Holtmann 			err = -EFAULT;
12161da177e4SLinus Torvalds 		break;
12171da177e4SLinus Torvalds 
12181da177e4SLinus Torvalds 	default:
1219cedc5469SMarcel Holtmann 		err = -ENOPROTOOPT;
12201da177e4SLinus Torvalds 		break;
12211da177e4SLinus Torvalds 	}
12221da177e4SLinus Torvalds 
1223cedc5469SMarcel Holtmann done:
1224cedc5469SMarcel Holtmann 	release_sock(sk);
1225cedc5469SMarcel Holtmann 	return err;
12261da177e4SLinus Torvalds }
12271da177e4SLinus Torvalds 
122890ddc4f0SEric Dumazet static const struct proto_ops hci_sock_ops = {
12291da177e4SLinus Torvalds 	.family		= PF_BLUETOOTH,
12301da177e4SLinus Torvalds 	.owner		= THIS_MODULE,
12311da177e4SLinus Torvalds 	.release	= hci_sock_release,
12321da177e4SLinus Torvalds 	.bind		= hci_sock_bind,
12331da177e4SLinus Torvalds 	.getname	= hci_sock_getname,
12341da177e4SLinus Torvalds 	.sendmsg	= hci_sock_sendmsg,
12351da177e4SLinus Torvalds 	.recvmsg	= hci_sock_recvmsg,
12361da177e4SLinus Torvalds 	.ioctl		= hci_sock_ioctl,
12371da177e4SLinus Torvalds 	.poll		= datagram_poll,
12381da177e4SLinus Torvalds 	.listen		= sock_no_listen,
12391da177e4SLinus Torvalds 	.shutdown	= sock_no_shutdown,
12401da177e4SLinus Torvalds 	.setsockopt	= hci_sock_setsockopt,
12411da177e4SLinus Torvalds 	.getsockopt	= hci_sock_getsockopt,
12421da177e4SLinus Torvalds 	.connect	= sock_no_connect,
12431da177e4SLinus Torvalds 	.socketpair	= sock_no_socketpair,
12441da177e4SLinus Torvalds 	.accept		= sock_no_accept,
12451da177e4SLinus Torvalds 	.mmap		= sock_no_mmap
12461da177e4SLinus Torvalds };
12471da177e4SLinus Torvalds 
12481da177e4SLinus Torvalds static struct proto hci_sk_proto = {
12491da177e4SLinus Torvalds 	.name		= "HCI",
12501da177e4SLinus Torvalds 	.owner		= THIS_MODULE,
12511da177e4SLinus Torvalds 	.obj_size	= sizeof(struct hci_pinfo)
12521da177e4SLinus Torvalds };
12531da177e4SLinus Torvalds 
12543f378b68SEric Paris static int hci_sock_create(struct net *net, struct socket *sock, int protocol,
12553f378b68SEric Paris 			   int kern)
12561da177e4SLinus Torvalds {
12571da177e4SLinus Torvalds 	struct sock *sk;
12581da177e4SLinus Torvalds 
12591da177e4SLinus Torvalds 	BT_DBG("sock %p", sock);
12601da177e4SLinus Torvalds 
12611da177e4SLinus Torvalds 	if (sock->type != SOCK_RAW)
12621da177e4SLinus Torvalds 		return -ESOCKTNOSUPPORT;
12631da177e4SLinus Torvalds 
12641da177e4SLinus Torvalds 	sock->ops = &hci_sock_ops;
12651da177e4SLinus Torvalds 
12666257ff21SPavel Emelyanov 	sk = sk_alloc(net, PF_BLUETOOTH, GFP_ATOMIC, &hci_sk_proto);
12671da177e4SLinus Torvalds 	if (!sk)
12681da177e4SLinus Torvalds 		return -ENOMEM;
12691da177e4SLinus Torvalds 
12701da177e4SLinus Torvalds 	sock_init_data(sock, sk);
12711da177e4SLinus Torvalds 
12721da177e4SLinus Torvalds 	sock_reset_flag(sk, SOCK_ZAPPED);
12731da177e4SLinus Torvalds 
12741da177e4SLinus Torvalds 	sk->sk_protocol = protocol;
12751da177e4SLinus Torvalds 
12761da177e4SLinus Torvalds 	sock->state = SS_UNCONNECTED;
12771da177e4SLinus Torvalds 	sk->sk_state = BT_OPEN;
12781da177e4SLinus Torvalds 
12791da177e4SLinus Torvalds 	bt_sock_link(&hci_sk_list, sk);
12801da177e4SLinus Torvalds 	return 0;
12811da177e4SLinus Torvalds }
12821da177e4SLinus Torvalds 
1283ec1b4cf7SStephen Hemminger static const struct net_proto_family hci_sock_family_ops = {
12841da177e4SLinus Torvalds 	.family	= PF_BLUETOOTH,
12851da177e4SLinus Torvalds 	.owner	= THIS_MODULE,
12861da177e4SLinus Torvalds 	.create	= hci_sock_create,
12871da177e4SLinus Torvalds };
12881da177e4SLinus Torvalds 
12891da177e4SLinus Torvalds int __init hci_sock_init(void)
12901da177e4SLinus Torvalds {
12911da177e4SLinus Torvalds 	int err;
12921da177e4SLinus Torvalds 
1293b0a8e282SMarcel Holtmann 	BUILD_BUG_ON(sizeof(struct sockaddr_hci) > sizeof(struct sockaddr));
1294b0a8e282SMarcel Holtmann 
12951da177e4SLinus Torvalds 	err = proto_register(&hci_sk_proto, 0);
12961da177e4SLinus Torvalds 	if (err < 0)
12971da177e4SLinus Torvalds 		return err;
12981da177e4SLinus Torvalds 
12991da177e4SLinus Torvalds 	err = bt_sock_register(BTPROTO_HCI, &hci_sock_family_ops);
1300f7c86637SMasatake YAMATO 	if (err < 0) {
1301f7c86637SMasatake YAMATO 		BT_ERR("HCI socket registration failed");
13021da177e4SLinus Torvalds 		goto error;
1303f7c86637SMasatake YAMATO 	}
1304f7c86637SMasatake YAMATO 
1305b0316615SAl Viro 	err = bt_procfs_init(&init_net, "hci", &hci_sk_list, NULL);
1306f7c86637SMasatake YAMATO 	if (err < 0) {
1307f7c86637SMasatake YAMATO 		BT_ERR("Failed to create HCI proc file");
1308f7c86637SMasatake YAMATO 		bt_sock_unregister(BTPROTO_HCI);
1309f7c86637SMasatake YAMATO 		goto error;
1310f7c86637SMasatake YAMATO 	}
13111da177e4SLinus Torvalds 
13121da177e4SLinus Torvalds 	BT_INFO("HCI socket layer initialized");
13131da177e4SLinus Torvalds 
13141da177e4SLinus Torvalds 	return 0;
13151da177e4SLinus Torvalds 
13161da177e4SLinus Torvalds error:
13171da177e4SLinus Torvalds 	proto_unregister(&hci_sk_proto);
13181da177e4SLinus Torvalds 	return err;
13191da177e4SLinus Torvalds }
13201da177e4SLinus Torvalds 
1321b7440a14SAnand Gadiyar void hci_sock_cleanup(void)
13221da177e4SLinus Torvalds {
1323f7c86637SMasatake YAMATO 	bt_procfs_cleanup(&init_net, "hci");
13245e9d7f86SDavid Herrmann 	bt_sock_unregister(BTPROTO_HCI);
13251da177e4SLinus Torvalds 	proto_unregister(&hci_sk_proto);
13261da177e4SLinus Torvalds }
1327