xref: /openbmc/linux/net/bluetooth/hci_sock.c (revision ed1b28a48b6c4e206bd88f5758393261710566f2)
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>
33fa4335d7SJohan Hedberg #include <net/bluetooth/mgmt.h>
34fa4335d7SJohan Hedberg 
35fa4335d7SJohan Hedberg #include "mgmt_util.h"
361da177e4SLinus Torvalds 
37801c1e8dSJohan Hedberg static LIST_HEAD(mgmt_chan_list);
38801c1e8dSJohan Hedberg static DEFINE_MUTEX(mgmt_chan_list_lock);
39801c1e8dSJohan Hedberg 
40cd82e61cSMarcel Holtmann static atomic_t monitor_promisc = ATOMIC_INIT(0);
41cd82e61cSMarcel Holtmann 
421da177e4SLinus Torvalds /* ----- HCI socket interface ----- */
431da177e4SLinus Torvalds 
44863def58SMarcel Holtmann /* Socket info */
45863def58SMarcel Holtmann #define hci_pi(sk) ((struct hci_pinfo *) sk)
46863def58SMarcel Holtmann 
47863def58SMarcel Holtmann struct hci_pinfo {
48863def58SMarcel Holtmann 	struct bt_sock    bt;
49863def58SMarcel Holtmann 	struct hci_dev    *hdev;
50863def58SMarcel Holtmann 	struct hci_filter filter;
51863def58SMarcel Holtmann 	__u32             cmsg_mask;
52863def58SMarcel Holtmann 	unsigned short    channel;
536befc644SMarcel Holtmann 	unsigned long     flags;
54863def58SMarcel Holtmann };
55863def58SMarcel Holtmann 
566befc644SMarcel Holtmann void hci_sock_set_flag(struct sock *sk, int nr)
576befc644SMarcel Holtmann {
586befc644SMarcel Holtmann 	set_bit(nr, &hci_pi(sk)->flags);
596befc644SMarcel Holtmann }
606befc644SMarcel Holtmann 
616befc644SMarcel Holtmann void hci_sock_clear_flag(struct sock *sk, int nr)
626befc644SMarcel Holtmann {
636befc644SMarcel Holtmann 	clear_bit(nr, &hci_pi(sk)->flags);
646befc644SMarcel Holtmann }
656befc644SMarcel Holtmann 
66c85be545SMarcel Holtmann int hci_sock_test_flag(struct sock *sk, int nr)
67c85be545SMarcel Holtmann {
68c85be545SMarcel Holtmann 	return test_bit(nr, &hci_pi(sk)->flags);
69c85be545SMarcel Holtmann }
70c85be545SMarcel Holtmann 
71d0f172b1SJohan Hedberg unsigned short hci_sock_get_channel(struct sock *sk)
72d0f172b1SJohan Hedberg {
73d0f172b1SJohan Hedberg 	return hci_pi(sk)->channel;
74d0f172b1SJohan Hedberg }
75d0f172b1SJohan Hedberg 
769391976aSJiri Slaby static inline int hci_test_bit(int nr, const void *addr)
771da177e4SLinus Torvalds {
789391976aSJiri Slaby 	return *((const __u32 *) addr + (nr >> 5)) & ((__u32) 1 << (nr & 31));
791da177e4SLinus Torvalds }
801da177e4SLinus Torvalds 
811da177e4SLinus Torvalds /* Security filter */
823ad254f7SMarcel Holtmann #define HCI_SFLT_MAX_OGF  5
833ad254f7SMarcel Holtmann 
843ad254f7SMarcel Holtmann struct hci_sec_filter {
853ad254f7SMarcel Holtmann 	__u32 type_mask;
863ad254f7SMarcel Holtmann 	__u32 event_mask[2];
873ad254f7SMarcel Holtmann 	__u32 ocf_mask[HCI_SFLT_MAX_OGF + 1][4];
883ad254f7SMarcel Holtmann };
893ad254f7SMarcel Holtmann 
907e67c112SMarcel Holtmann static const struct hci_sec_filter hci_sec_filter = {
911da177e4SLinus Torvalds 	/* Packet types */
921da177e4SLinus Torvalds 	0x10,
931da177e4SLinus Torvalds 	/* Events */
94dd7f5527SMarcel Holtmann 	{ 0x1000d9fe, 0x0000b00c },
951da177e4SLinus Torvalds 	/* Commands */
961da177e4SLinus Torvalds 	{
971da177e4SLinus Torvalds 		{ 0x0 },
981da177e4SLinus Torvalds 		/* OGF_LINK_CTL */
997c631a67SMarcel Holtmann 		{ 0xbe000006, 0x00000001, 0x00000000, 0x00 },
1001da177e4SLinus Torvalds 		/* OGF_LINK_POLICY */
1017c631a67SMarcel Holtmann 		{ 0x00005200, 0x00000000, 0x00000000, 0x00 },
1021da177e4SLinus Torvalds 		/* OGF_HOST_CTL */
1037c631a67SMarcel Holtmann 		{ 0xaab00200, 0x2b402aaa, 0x05220154, 0x00 },
1041da177e4SLinus Torvalds 		/* OGF_INFO_PARAM */
1057c631a67SMarcel Holtmann 		{ 0x000002be, 0x00000000, 0x00000000, 0x00 },
1061da177e4SLinus Torvalds 		/* OGF_STATUS_PARAM */
1077c631a67SMarcel Holtmann 		{ 0x000000ea, 0x00000000, 0x00000000, 0x00 }
1081da177e4SLinus Torvalds 	}
1091da177e4SLinus Torvalds };
1101da177e4SLinus Torvalds 
1111da177e4SLinus Torvalds static struct bt_sock_list hci_sk_list = {
112d5fb2962SRobert P. J. Day 	.lock = __RW_LOCK_UNLOCKED(hci_sk_list.lock)
1131da177e4SLinus Torvalds };
1141da177e4SLinus Torvalds 
115f81fe64fSMarcel Holtmann static bool is_filtered_packet(struct sock *sk, struct sk_buff *skb)
116f81fe64fSMarcel Holtmann {
117f81fe64fSMarcel Holtmann 	struct hci_filter *flt;
118f81fe64fSMarcel Holtmann 	int flt_type, flt_event;
119f81fe64fSMarcel Holtmann 
120f81fe64fSMarcel Holtmann 	/* Apply filter */
121f81fe64fSMarcel Holtmann 	flt = &hci_pi(sk)->filter;
122f81fe64fSMarcel Holtmann 
123f81fe64fSMarcel Holtmann 	if (bt_cb(skb)->pkt_type == HCI_VENDOR_PKT)
124f81fe64fSMarcel Holtmann 		flt_type = 0;
125f81fe64fSMarcel Holtmann 	else
126f81fe64fSMarcel Holtmann 		flt_type = bt_cb(skb)->pkt_type & HCI_FLT_TYPE_BITS;
127f81fe64fSMarcel Holtmann 
128f81fe64fSMarcel Holtmann 	if (!test_bit(flt_type, &flt->type_mask))
129f81fe64fSMarcel Holtmann 		return true;
130f81fe64fSMarcel Holtmann 
131f81fe64fSMarcel Holtmann 	/* Extra filter for event packets only */
132f81fe64fSMarcel Holtmann 	if (bt_cb(skb)->pkt_type != HCI_EVENT_PKT)
133f81fe64fSMarcel Holtmann 		return false;
134f81fe64fSMarcel Holtmann 
135f81fe64fSMarcel Holtmann 	flt_event = (*(__u8 *)skb->data & HCI_FLT_EVENT_BITS);
136f81fe64fSMarcel Holtmann 
137f81fe64fSMarcel Holtmann 	if (!hci_test_bit(flt_event, &flt->event_mask))
138f81fe64fSMarcel Holtmann 		return true;
139f81fe64fSMarcel Holtmann 
140f81fe64fSMarcel Holtmann 	/* Check filter only when opcode is set */
141f81fe64fSMarcel Holtmann 	if (!flt->opcode)
142f81fe64fSMarcel Holtmann 		return false;
143f81fe64fSMarcel Holtmann 
144f81fe64fSMarcel Holtmann 	if (flt_event == HCI_EV_CMD_COMPLETE &&
145f81fe64fSMarcel Holtmann 	    flt->opcode != get_unaligned((__le16 *)(skb->data + 3)))
146f81fe64fSMarcel Holtmann 		return true;
147f81fe64fSMarcel Holtmann 
148f81fe64fSMarcel Holtmann 	if (flt_event == HCI_EV_CMD_STATUS &&
149f81fe64fSMarcel Holtmann 	    flt->opcode != get_unaligned((__le16 *)(skb->data + 4)))
150f81fe64fSMarcel Holtmann 		return true;
151f81fe64fSMarcel Holtmann 
152f81fe64fSMarcel Holtmann 	return false;
153f81fe64fSMarcel Holtmann }
154f81fe64fSMarcel Holtmann 
1551da177e4SLinus Torvalds /* Send frame to RAW socket */
156470fe1b5SMarcel Holtmann void hci_send_to_sock(struct hci_dev *hdev, struct sk_buff *skb)
1571da177e4SLinus Torvalds {
1581da177e4SLinus Torvalds 	struct sock *sk;
159e0edf373SMarcel Holtmann 	struct sk_buff *skb_copy = NULL;
1601da177e4SLinus Torvalds 
1611da177e4SLinus Torvalds 	BT_DBG("hdev %p len %d", hdev, skb->len);
1621da177e4SLinus Torvalds 
1631da177e4SLinus Torvalds 	read_lock(&hci_sk_list.lock);
164470fe1b5SMarcel Holtmann 
165b67bfe0dSSasha Levin 	sk_for_each(sk, &hci_sk_list.head) {
1661da177e4SLinus Torvalds 		struct sk_buff *nskb;
1671da177e4SLinus Torvalds 
1681da177e4SLinus Torvalds 		if (sk->sk_state != BT_BOUND || hci_pi(sk)->hdev != hdev)
1691da177e4SLinus Torvalds 			continue;
1701da177e4SLinus Torvalds 
1711da177e4SLinus Torvalds 		/* Don't send frame to the socket it came from */
1721da177e4SLinus Torvalds 		if (skb->sk == sk)
1731da177e4SLinus Torvalds 			continue;
1741da177e4SLinus Torvalds 
17523500189SMarcel Holtmann 		if (hci_pi(sk)->channel == HCI_CHANNEL_RAW) {
176f81fe64fSMarcel Holtmann 			if (is_filtered_packet(sk, skb))
1771da177e4SLinus Torvalds 				continue;
17823500189SMarcel Holtmann 		} else if (hci_pi(sk)->channel == HCI_CHANNEL_USER) {
17923500189SMarcel Holtmann 			if (!bt_cb(skb)->incoming)
18023500189SMarcel Holtmann 				continue;
18123500189SMarcel Holtmann 			if (bt_cb(skb)->pkt_type != HCI_EVENT_PKT &&
18223500189SMarcel Holtmann 			    bt_cb(skb)->pkt_type != HCI_ACLDATA_PKT &&
18323500189SMarcel Holtmann 			    bt_cb(skb)->pkt_type != HCI_SCODATA_PKT)
18423500189SMarcel Holtmann 				continue;
18523500189SMarcel Holtmann 		} else {
18623500189SMarcel Holtmann 			/* Don't send frame to other channel types */
18723500189SMarcel Holtmann 			continue;
18823500189SMarcel Holtmann 		}
1891da177e4SLinus Torvalds 
190e0edf373SMarcel Holtmann 		if (!skb_copy) {
191e0edf373SMarcel Holtmann 			/* Create a private copy with headroom */
192bad93e9dSOctavian Purdila 			skb_copy = __pskb_copy_fclone(skb, 1, GFP_ATOMIC, true);
193e0edf373SMarcel Holtmann 			if (!skb_copy)
1941da177e4SLinus Torvalds 				continue;
1951da177e4SLinus Torvalds 
1961da177e4SLinus Torvalds 			/* Put type byte before the data */
197e0edf373SMarcel Holtmann 			memcpy(skb_push(skb_copy, 1), &bt_cb(skb)->pkt_type, 1);
198e0edf373SMarcel Holtmann 		}
199e0edf373SMarcel Holtmann 
200e0edf373SMarcel Holtmann 		nskb = skb_clone(skb_copy, GFP_ATOMIC);
201e0edf373SMarcel Holtmann 		if (!nskb)
202e0edf373SMarcel Holtmann 			continue;
2031da177e4SLinus Torvalds 
2041da177e4SLinus Torvalds 		if (sock_queue_rcv_skb(sk, nskb))
2051da177e4SLinus Torvalds 			kfree_skb(nskb);
2061da177e4SLinus Torvalds 	}
207470fe1b5SMarcel Holtmann 
208470fe1b5SMarcel Holtmann 	read_unlock(&hci_sk_list.lock);
209e0edf373SMarcel Holtmann 
210e0edf373SMarcel Holtmann 	kfree_skb(skb_copy);
211470fe1b5SMarcel Holtmann }
212470fe1b5SMarcel Holtmann 
2137129069eSJohan Hedberg /* Send frame to sockets with specific channel */
2147129069eSJohan Hedberg void hci_send_to_channel(unsigned short channel, struct sk_buff *skb,
215c08b1a1dSMarcel Holtmann 			 int flag, struct sock *skip_sk)
216470fe1b5SMarcel Holtmann {
217470fe1b5SMarcel Holtmann 	struct sock *sk;
218470fe1b5SMarcel Holtmann 
2197129069eSJohan Hedberg 	BT_DBG("channel %u len %d", channel, skb->len);
220470fe1b5SMarcel Holtmann 
221470fe1b5SMarcel Holtmann 	read_lock(&hci_sk_list.lock);
222470fe1b5SMarcel Holtmann 
223b67bfe0dSSasha Levin 	sk_for_each(sk, &hci_sk_list.head) {
224470fe1b5SMarcel Holtmann 		struct sk_buff *nskb;
225470fe1b5SMarcel Holtmann 
226c08b1a1dSMarcel Holtmann 		/* Ignore socket without the flag set */
227c85be545SMarcel Holtmann 		if (!hci_sock_test_flag(sk, flag))
228c08b1a1dSMarcel Holtmann 			continue;
229c08b1a1dSMarcel Holtmann 
230470fe1b5SMarcel Holtmann 		/* Skip the original socket */
231470fe1b5SMarcel Holtmann 		if (sk == skip_sk)
232470fe1b5SMarcel Holtmann 			continue;
233470fe1b5SMarcel Holtmann 
234470fe1b5SMarcel Holtmann 		if (sk->sk_state != BT_BOUND)
235470fe1b5SMarcel Holtmann 			continue;
236470fe1b5SMarcel Holtmann 
2377129069eSJohan Hedberg 		if (hci_pi(sk)->channel != channel)
238d7f72f61SMarcel Holtmann 			continue;
239d7f72f61SMarcel Holtmann 
240d7f72f61SMarcel Holtmann 		nskb = skb_clone(skb, GFP_ATOMIC);
241d7f72f61SMarcel Holtmann 		if (!nskb)
242d7f72f61SMarcel Holtmann 			continue;
243d7f72f61SMarcel Holtmann 
244d7f72f61SMarcel Holtmann 		if (sock_queue_rcv_skb(sk, nskb))
245d7f72f61SMarcel Holtmann 			kfree_skb(nskb);
246d7f72f61SMarcel Holtmann 	}
247d7f72f61SMarcel Holtmann 
248d7f72f61SMarcel Holtmann 	read_unlock(&hci_sk_list.lock);
249d7f72f61SMarcel Holtmann }
250d7f72f61SMarcel Holtmann 
251cd82e61cSMarcel Holtmann /* Send frame to monitor socket */
252cd82e61cSMarcel Holtmann void hci_send_to_monitor(struct hci_dev *hdev, struct sk_buff *skb)
253cd82e61cSMarcel Holtmann {
254cd82e61cSMarcel Holtmann 	struct sk_buff *skb_copy = NULL;
2552b531294SMarcel Holtmann 	struct hci_mon_hdr *hdr;
256cd82e61cSMarcel Holtmann 	__le16 opcode;
257cd82e61cSMarcel Holtmann 
258cd82e61cSMarcel Holtmann 	if (!atomic_read(&monitor_promisc))
259cd82e61cSMarcel Holtmann 		return;
260cd82e61cSMarcel Holtmann 
261cd82e61cSMarcel Holtmann 	BT_DBG("hdev %p len %d", hdev, skb->len);
262cd82e61cSMarcel Holtmann 
263cd82e61cSMarcel Holtmann 	switch (bt_cb(skb)->pkt_type) {
264cd82e61cSMarcel Holtmann 	case HCI_COMMAND_PKT:
265dcf4adbfSJoe Perches 		opcode = cpu_to_le16(HCI_MON_COMMAND_PKT);
266cd82e61cSMarcel Holtmann 		break;
267cd82e61cSMarcel Holtmann 	case HCI_EVENT_PKT:
268dcf4adbfSJoe Perches 		opcode = cpu_to_le16(HCI_MON_EVENT_PKT);
269cd82e61cSMarcel Holtmann 		break;
270cd82e61cSMarcel Holtmann 	case HCI_ACLDATA_PKT:
271cd82e61cSMarcel Holtmann 		if (bt_cb(skb)->incoming)
272dcf4adbfSJoe Perches 			opcode = cpu_to_le16(HCI_MON_ACL_RX_PKT);
273cd82e61cSMarcel Holtmann 		else
274dcf4adbfSJoe Perches 			opcode = cpu_to_le16(HCI_MON_ACL_TX_PKT);
275cd82e61cSMarcel Holtmann 		break;
276cd82e61cSMarcel Holtmann 	case HCI_SCODATA_PKT:
277cd82e61cSMarcel Holtmann 		if (bt_cb(skb)->incoming)
278dcf4adbfSJoe Perches 			opcode = cpu_to_le16(HCI_MON_SCO_RX_PKT);
279cd82e61cSMarcel Holtmann 		else
280dcf4adbfSJoe Perches 			opcode = cpu_to_le16(HCI_MON_SCO_TX_PKT);
281cd82e61cSMarcel Holtmann 		break;
282cd82e61cSMarcel Holtmann 	default:
283cd82e61cSMarcel Holtmann 		return;
284cd82e61cSMarcel Holtmann 	}
285cd82e61cSMarcel Holtmann 
2862b531294SMarcel Holtmann 	/* Create a private copy with headroom */
2872b531294SMarcel Holtmann 	skb_copy = __pskb_copy_fclone(skb, HCI_MON_HDR_SIZE, GFP_ATOMIC, true);
2882b531294SMarcel Holtmann 	if (!skb_copy)
2892b531294SMarcel Holtmann 		return;
2902b531294SMarcel Holtmann 
2912b531294SMarcel Holtmann 	/* Put header before the data */
2922b531294SMarcel Holtmann 	hdr = (void *) skb_push(skb_copy, HCI_MON_HDR_SIZE);
2932b531294SMarcel Holtmann 	hdr->opcode = opcode;
2942b531294SMarcel Holtmann 	hdr->index = cpu_to_le16(hdev->id);
2952b531294SMarcel Holtmann 	hdr->len = cpu_to_le16(skb->len);
2962b531294SMarcel Holtmann 
297c08b1a1dSMarcel Holtmann 	hci_send_to_channel(HCI_CHANNEL_MONITOR, skb_copy,
298c08b1a1dSMarcel Holtmann 			    HCI_SOCK_TRUSTED, NULL);
299cd82e61cSMarcel Holtmann 	kfree_skb(skb_copy);
300cd82e61cSMarcel Holtmann }
301cd82e61cSMarcel Holtmann 
302cd82e61cSMarcel Holtmann static struct sk_buff *create_monitor_event(struct hci_dev *hdev, int event)
303cd82e61cSMarcel Holtmann {
304cd82e61cSMarcel Holtmann 	struct hci_mon_hdr *hdr;
305cd82e61cSMarcel Holtmann 	struct hci_mon_new_index *ni;
306cd82e61cSMarcel Holtmann 	struct sk_buff *skb;
307cd82e61cSMarcel Holtmann 	__le16 opcode;
308cd82e61cSMarcel Holtmann 
309cd82e61cSMarcel Holtmann 	switch (event) {
310cd82e61cSMarcel Holtmann 	case HCI_DEV_REG:
311cd82e61cSMarcel Holtmann 		skb = bt_skb_alloc(HCI_MON_NEW_INDEX_SIZE, GFP_ATOMIC);
312cd82e61cSMarcel Holtmann 		if (!skb)
313cd82e61cSMarcel Holtmann 			return NULL;
314cd82e61cSMarcel Holtmann 
315cd82e61cSMarcel Holtmann 		ni = (void *) skb_put(skb, HCI_MON_NEW_INDEX_SIZE);
316cd82e61cSMarcel Holtmann 		ni->type = hdev->dev_type;
317cd82e61cSMarcel Holtmann 		ni->bus = hdev->bus;
318cd82e61cSMarcel Holtmann 		bacpy(&ni->bdaddr, &hdev->bdaddr);
319cd82e61cSMarcel Holtmann 		memcpy(ni->name, hdev->name, 8);
320cd82e61cSMarcel Holtmann 
321dcf4adbfSJoe Perches 		opcode = cpu_to_le16(HCI_MON_NEW_INDEX);
322cd82e61cSMarcel Holtmann 		break;
323cd82e61cSMarcel Holtmann 
324cd82e61cSMarcel Holtmann 	case HCI_DEV_UNREG:
325cd82e61cSMarcel Holtmann 		skb = bt_skb_alloc(0, GFP_ATOMIC);
326cd82e61cSMarcel Holtmann 		if (!skb)
327cd82e61cSMarcel Holtmann 			return NULL;
328cd82e61cSMarcel Holtmann 
329dcf4adbfSJoe Perches 		opcode = cpu_to_le16(HCI_MON_DEL_INDEX);
330cd82e61cSMarcel Holtmann 		break;
331cd82e61cSMarcel Holtmann 
332cd82e61cSMarcel Holtmann 	default:
333cd82e61cSMarcel Holtmann 		return NULL;
334cd82e61cSMarcel Holtmann 	}
335cd82e61cSMarcel Holtmann 
336cd82e61cSMarcel Holtmann 	__net_timestamp(skb);
337cd82e61cSMarcel Holtmann 
338cd82e61cSMarcel Holtmann 	hdr = (void *) skb_push(skb, HCI_MON_HDR_SIZE);
339cd82e61cSMarcel Holtmann 	hdr->opcode = opcode;
340cd82e61cSMarcel Holtmann 	hdr->index = cpu_to_le16(hdev->id);
341cd82e61cSMarcel Holtmann 	hdr->len = cpu_to_le16(skb->len - HCI_MON_HDR_SIZE);
342cd82e61cSMarcel Holtmann 
343cd82e61cSMarcel Holtmann 	return skb;
344cd82e61cSMarcel Holtmann }
345cd82e61cSMarcel Holtmann 
346cd82e61cSMarcel Holtmann static void send_monitor_replay(struct sock *sk)
347cd82e61cSMarcel Holtmann {
348cd82e61cSMarcel Holtmann 	struct hci_dev *hdev;
349cd82e61cSMarcel Holtmann 
350cd82e61cSMarcel Holtmann 	read_lock(&hci_dev_list_lock);
351cd82e61cSMarcel Holtmann 
352cd82e61cSMarcel Holtmann 	list_for_each_entry(hdev, &hci_dev_list, list) {
353cd82e61cSMarcel Holtmann 		struct sk_buff *skb;
354cd82e61cSMarcel Holtmann 
355cd82e61cSMarcel Holtmann 		skb = create_monitor_event(hdev, HCI_DEV_REG);
356cd82e61cSMarcel Holtmann 		if (!skb)
357cd82e61cSMarcel Holtmann 			continue;
358cd82e61cSMarcel Holtmann 
359cd82e61cSMarcel Holtmann 		if (sock_queue_rcv_skb(sk, skb))
360cd82e61cSMarcel Holtmann 			kfree_skb(skb);
361cd82e61cSMarcel Holtmann 	}
362cd82e61cSMarcel Holtmann 
363cd82e61cSMarcel Holtmann 	read_unlock(&hci_dev_list_lock);
364cd82e61cSMarcel Holtmann }
365cd82e61cSMarcel Holtmann 
366040030efSMarcel Holtmann /* Generate internal stack event */
367040030efSMarcel Holtmann static void hci_si_event(struct hci_dev *hdev, int type, int dlen, void *data)
368040030efSMarcel Holtmann {
369040030efSMarcel Holtmann 	struct hci_event_hdr *hdr;
370040030efSMarcel Holtmann 	struct hci_ev_stack_internal *ev;
371040030efSMarcel Holtmann 	struct sk_buff *skb;
372040030efSMarcel Holtmann 
373040030efSMarcel Holtmann 	skb = bt_skb_alloc(HCI_EVENT_HDR_SIZE + sizeof(*ev) + dlen, GFP_ATOMIC);
374040030efSMarcel Holtmann 	if (!skb)
375040030efSMarcel Holtmann 		return;
376040030efSMarcel Holtmann 
377040030efSMarcel Holtmann 	hdr = (void *) skb_put(skb, HCI_EVENT_HDR_SIZE);
378040030efSMarcel Holtmann 	hdr->evt  = HCI_EV_STACK_INTERNAL;
379040030efSMarcel Holtmann 	hdr->plen = sizeof(*ev) + dlen;
380040030efSMarcel Holtmann 
381040030efSMarcel Holtmann 	ev  = (void *) skb_put(skb, sizeof(*ev) + dlen);
382040030efSMarcel Holtmann 	ev->type = type;
383040030efSMarcel Holtmann 	memcpy(ev->data, data, dlen);
384040030efSMarcel Holtmann 
385040030efSMarcel Holtmann 	bt_cb(skb)->incoming = 1;
386040030efSMarcel Holtmann 	__net_timestamp(skb);
387040030efSMarcel Holtmann 
388040030efSMarcel Holtmann 	bt_cb(skb)->pkt_type = HCI_EVENT_PKT;
389040030efSMarcel Holtmann 	hci_send_to_sock(hdev, skb);
390040030efSMarcel Holtmann 	kfree_skb(skb);
391040030efSMarcel Holtmann }
392040030efSMarcel Holtmann 
393040030efSMarcel Holtmann void hci_sock_dev_event(struct hci_dev *hdev, int event)
394040030efSMarcel Holtmann {
395040030efSMarcel Holtmann 	BT_DBG("hdev %s event %d", hdev->name, event);
396040030efSMarcel Holtmann 
397cd82e61cSMarcel Holtmann 	if (atomic_read(&monitor_promisc)) {
398cd82e61cSMarcel Holtmann 		struct sk_buff *skb;
399cd82e61cSMarcel Holtmann 
400*ed1b28a4SMarcel Holtmann 		/* Send event to monitor */
401cd82e61cSMarcel Holtmann 		skb = create_monitor_event(hdev, event);
402cd82e61cSMarcel Holtmann 		if (skb) {
403c08b1a1dSMarcel Holtmann 			hci_send_to_channel(HCI_CHANNEL_MONITOR, skb,
404c08b1a1dSMarcel Holtmann 					    HCI_SOCK_TRUSTED, NULL);
405cd82e61cSMarcel Holtmann 			kfree_skb(skb);
406cd82e61cSMarcel Holtmann 		}
407cd82e61cSMarcel Holtmann 	}
408cd82e61cSMarcel Holtmann 
409*ed1b28a4SMarcel Holtmann 	if (event <= HCI_DEV_DOWN) {
410*ed1b28a4SMarcel Holtmann 		struct hci_ev_si_device ev;
411*ed1b28a4SMarcel Holtmann 
412040030efSMarcel Holtmann 		/* Send event to sockets */
413040030efSMarcel Holtmann 		ev.event  = event;
414040030efSMarcel Holtmann 		ev.dev_id = hdev->id;
415040030efSMarcel Holtmann 		hci_si_event(NULL, HCI_EV_SI_DEVICE, sizeof(ev), &ev);
416*ed1b28a4SMarcel Holtmann 	}
417040030efSMarcel Holtmann 
418040030efSMarcel Holtmann 	if (event == HCI_DEV_UNREG) {
419040030efSMarcel Holtmann 		struct sock *sk;
420040030efSMarcel Holtmann 
421040030efSMarcel Holtmann 		/* Detach sockets from device */
422040030efSMarcel Holtmann 		read_lock(&hci_sk_list.lock);
423b67bfe0dSSasha Levin 		sk_for_each(sk, &hci_sk_list.head) {
424040030efSMarcel Holtmann 			bh_lock_sock_nested(sk);
425040030efSMarcel Holtmann 			if (hci_pi(sk)->hdev == hdev) {
426040030efSMarcel Holtmann 				hci_pi(sk)->hdev = NULL;
427040030efSMarcel Holtmann 				sk->sk_err = EPIPE;
428040030efSMarcel Holtmann 				sk->sk_state = BT_OPEN;
429040030efSMarcel Holtmann 				sk->sk_state_change(sk);
430040030efSMarcel Holtmann 
431040030efSMarcel Holtmann 				hci_dev_put(hdev);
432040030efSMarcel Holtmann 			}
433040030efSMarcel Holtmann 			bh_unlock_sock(sk);
434040030efSMarcel Holtmann 		}
435040030efSMarcel Holtmann 		read_unlock(&hci_sk_list.lock);
436040030efSMarcel Holtmann 	}
437040030efSMarcel Holtmann }
438040030efSMarcel Holtmann 
439801c1e8dSJohan Hedberg static struct hci_mgmt_chan *__hci_mgmt_chan_find(unsigned short channel)
440801c1e8dSJohan Hedberg {
441801c1e8dSJohan Hedberg 	struct hci_mgmt_chan *c;
442801c1e8dSJohan Hedberg 
443801c1e8dSJohan Hedberg 	list_for_each_entry(c, &mgmt_chan_list, list) {
444801c1e8dSJohan Hedberg 		if (c->channel == channel)
445801c1e8dSJohan Hedberg 			return c;
446801c1e8dSJohan Hedberg 	}
447801c1e8dSJohan Hedberg 
448801c1e8dSJohan Hedberg 	return NULL;
449801c1e8dSJohan Hedberg }
450801c1e8dSJohan Hedberg 
451801c1e8dSJohan Hedberg static struct hci_mgmt_chan *hci_mgmt_chan_find(unsigned short channel)
452801c1e8dSJohan Hedberg {
453801c1e8dSJohan Hedberg 	struct hci_mgmt_chan *c;
454801c1e8dSJohan Hedberg 
455801c1e8dSJohan Hedberg 	mutex_lock(&mgmt_chan_list_lock);
456801c1e8dSJohan Hedberg 	c = __hci_mgmt_chan_find(channel);
457801c1e8dSJohan Hedberg 	mutex_unlock(&mgmt_chan_list_lock);
458801c1e8dSJohan Hedberg 
459801c1e8dSJohan Hedberg 	return c;
460801c1e8dSJohan Hedberg }
461801c1e8dSJohan Hedberg 
462801c1e8dSJohan Hedberg int hci_mgmt_chan_register(struct hci_mgmt_chan *c)
463801c1e8dSJohan Hedberg {
464801c1e8dSJohan Hedberg 	if (c->channel < HCI_CHANNEL_CONTROL)
465801c1e8dSJohan Hedberg 		return -EINVAL;
466801c1e8dSJohan Hedberg 
467801c1e8dSJohan Hedberg 	mutex_lock(&mgmt_chan_list_lock);
468801c1e8dSJohan Hedberg 	if (__hci_mgmt_chan_find(c->channel)) {
469801c1e8dSJohan Hedberg 		mutex_unlock(&mgmt_chan_list_lock);
470801c1e8dSJohan Hedberg 		return -EALREADY;
471801c1e8dSJohan Hedberg 	}
472801c1e8dSJohan Hedberg 
473801c1e8dSJohan Hedberg 	list_add_tail(&c->list, &mgmt_chan_list);
474801c1e8dSJohan Hedberg 
475801c1e8dSJohan Hedberg 	mutex_unlock(&mgmt_chan_list_lock);
476801c1e8dSJohan Hedberg 
477801c1e8dSJohan Hedberg 	return 0;
478801c1e8dSJohan Hedberg }
479801c1e8dSJohan Hedberg EXPORT_SYMBOL(hci_mgmt_chan_register);
480801c1e8dSJohan Hedberg 
481801c1e8dSJohan Hedberg void hci_mgmt_chan_unregister(struct hci_mgmt_chan *c)
482801c1e8dSJohan Hedberg {
483801c1e8dSJohan Hedberg 	mutex_lock(&mgmt_chan_list_lock);
484801c1e8dSJohan Hedberg 	list_del(&c->list);
485801c1e8dSJohan Hedberg 	mutex_unlock(&mgmt_chan_list_lock);
486801c1e8dSJohan Hedberg }
487801c1e8dSJohan Hedberg EXPORT_SYMBOL(hci_mgmt_chan_unregister);
488801c1e8dSJohan Hedberg 
4891da177e4SLinus Torvalds static int hci_sock_release(struct socket *sock)
4901da177e4SLinus Torvalds {
4911da177e4SLinus Torvalds 	struct sock *sk = sock->sk;
4927b005bd3SMarcel Holtmann 	struct hci_dev *hdev;
4931da177e4SLinus Torvalds 
4941da177e4SLinus Torvalds 	BT_DBG("sock %p sk %p", sock, sk);
4951da177e4SLinus Torvalds 
4961da177e4SLinus Torvalds 	if (!sk)
4971da177e4SLinus Torvalds 		return 0;
4981da177e4SLinus Torvalds 
4997b005bd3SMarcel Holtmann 	hdev = hci_pi(sk)->hdev;
5007b005bd3SMarcel Holtmann 
501cd82e61cSMarcel Holtmann 	if (hci_pi(sk)->channel == HCI_CHANNEL_MONITOR)
502cd82e61cSMarcel Holtmann 		atomic_dec(&monitor_promisc);
503cd82e61cSMarcel Holtmann 
5041da177e4SLinus Torvalds 	bt_sock_unlink(&hci_sk_list, sk);
5051da177e4SLinus Torvalds 
5061da177e4SLinus Torvalds 	if (hdev) {
50723500189SMarcel Holtmann 		if (hci_pi(sk)->channel == HCI_CHANNEL_USER) {
5086b3cc1dbSSimon Fels 			/* When releasing an user channel exclusive access,
5096b3cc1dbSSimon Fels 			 * call hci_dev_do_close directly instead of calling
5106b3cc1dbSSimon Fels 			 * hci_dev_close to ensure the exclusive access will
5116b3cc1dbSSimon Fels 			 * be released and the controller brought back down.
5126b3cc1dbSSimon Fels 			 *
5136b3cc1dbSSimon Fels 			 * The checking of HCI_AUTO_OFF is not needed in this
5146b3cc1dbSSimon Fels 			 * case since it will have been cleared already when
5156b3cc1dbSSimon Fels 			 * opening the user channel.
5166b3cc1dbSSimon Fels 			 */
5176b3cc1dbSSimon Fels 			hci_dev_do_close(hdev);
5189380f9eaSLoic Poulain 			hci_dev_clear_flag(hdev, HCI_USER_CHANNEL);
5199380f9eaSLoic Poulain 			mgmt_index_added(hdev);
52023500189SMarcel Holtmann 		}
52123500189SMarcel Holtmann 
5221da177e4SLinus Torvalds 		atomic_dec(&hdev->promisc);
5231da177e4SLinus Torvalds 		hci_dev_put(hdev);
5241da177e4SLinus Torvalds 	}
5251da177e4SLinus Torvalds 
5261da177e4SLinus Torvalds 	sock_orphan(sk);
5271da177e4SLinus Torvalds 
5281da177e4SLinus Torvalds 	skb_queue_purge(&sk->sk_receive_queue);
5291da177e4SLinus Torvalds 	skb_queue_purge(&sk->sk_write_queue);
5301da177e4SLinus Torvalds 
5311da177e4SLinus Torvalds 	sock_put(sk);
5321da177e4SLinus Torvalds 	return 0;
5331da177e4SLinus Torvalds }
5341da177e4SLinus Torvalds 
535b2a66aadSAntti Julku static int hci_sock_blacklist_add(struct hci_dev *hdev, void __user *arg)
536f0358568SJohan Hedberg {
537f0358568SJohan Hedberg 	bdaddr_t bdaddr;
5385e762444SAntti Julku 	int err;
539f0358568SJohan Hedberg 
540f0358568SJohan Hedberg 	if (copy_from_user(&bdaddr, arg, sizeof(bdaddr)))
541f0358568SJohan Hedberg 		return -EFAULT;
542f0358568SJohan Hedberg 
54309fd0de5SGustavo F. Padovan 	hci_dev_lock(hdev);
5445e762444SAntti Julku 
545dcc36c16SJohan Hedberg 	err = hci_bdaddr_list_add(&hdev->blacklist, &bdaddr, BDADDR_BREDR);
5465e762444SAntti Julku 
54709fd0de5SGustavo F. Padovan 	hci_dev_unlock(hdev);
5485e762444SAntti Julku 
5495e762444SAntti Julku 	return err;
550f0358568SJohan Hedberg }
551f0358568SJohan Hedberg 
552b2a66aadSAntti Julku static int hci_sock_blacklist_del(struct hci_dev *hdev, void __user *arg)
553f0358568SJohan Hedberg {
554f0358568SJohan Hedberg 	bdaddr_t bdaddr;
5555e762444SAntti Julku 	int err;
556f0358568SJohan Hedberg 
557f0358568SJohan Hedberg 	if (copy_from_user(&bdaddr, arg, sizeof(bdaddr)))
558f0358568SJohan Hedberg 		return -EFAULT;
559f0358568SJohan Hedberg 
56009fd0de5SGustavo F. Padovan 	hci_dev_lock(hdev);
5615e762444SAntti Julku 
562dcc36c16SJohan Hedberg 	err = hci_bdaddr_list_del(&hdev->blacklist, &bdaddr, BDADDR_BREDR);
5635e762444SAntti Julku 
56409fd0de5SGustavo F. Padovan 	hci_dev_unlock(hdev);
5655e762444SAntti Julku 
5665e762444SAntti Julku 	return err;
567f0358568SJohan Hedberg }
568f0358568SJohan Hedberg 
5691da177e4SLinus Torvalds /* Ioctls that require bound socket */
5706039aa73SGustavo Padovan static int hci_sock_bound_ioctl(struct sock *sk, unsigned int cmd,
5716039aa73SGustavo Padovan 				unsigned long arg)
5721da177e4SLinus Torvalds {
5731da177e4SLinus Torvalds 	struct hci_dev *hdev = hci_pi(sk)->hdev;
5741da177e4SLinus Torvalds 
5751da177e4SLinus Torvalds 	if (!hdev)
5761da177e4SLinus Torvalds 		return -EBADFD;
5771da177e4SLinus Torvalds 
578d7a5a11dSMarcel Holtmann 	if (hci_dev_test_flag(hdev, HCI_USER_CHANNEL))
5790736cfa8SMarcel Holtmann 		return -EBUSY;
5800736cfa8SMarcel Holtmann 
581d7a5a11dSMarcel Holtmann 	if (hci_dev_test_flag(hdev, HCI_UNCONFIGURED))
582fee746b0SMarcel Holtmann 		return -EOPNOTSUPP;
583fee746b0SMarcel Holtmann 
5845b69bef5SMarcel Holtmann 	if (hdev->dev_type != HCI_BREDR)
5855b69bef5SMarcel Holtmann 		return -EOPNOTSUPP;
5865b69bef5SMarcel Holtmann 
5871da177e4SLinus Torvalds 	switch (cmd) {
5881da177e4SLinus Torvalds 	case HCISETRAW:
5891da177e4SLinus Torvalds 		if (!capable(CAP_NET_ADMIN))
590bf5b30b8SZhao Hongjiang 			return -EPERM;
591db596681SMarcel Holtmann 		return -EOPNOTSUPP;
5921da177e4SLinus Torvalds 
5931da177e4SLinus Torvalds 	case HCIGETCONNINFO:
5941da177e4SLinus Torvalds 		return hci_get_conn_info(hdev, (void __user *) arg);
5951da177e4SLinus Torvalds 
59640be492fSMarcel Holtmann 	case HCIGETAUTHINFO:
59740be492fSMarcel Holtmann 		return hci_get_auth_info(hdev, (void __user *) arg);
59840be492fSMarcel Holtmann 
599f0358568SJohan Hedberg 	case HCIBLOCKADDR:
600f0358568SJohan Hedberg 		if (!capable(CAP_NET_ADMIN))
601bf5b30b8SZhao Hongjiang 			return -EPERM;
602b2a66aadSAntti Julku 		return hci_sock_blacklist_add(hdev, (void __user *) arg);
603f0358568SJohan Hedberg 
604f0358568SJohan Hedberg 	case HCIUNBLOCKADDR:
605f0358568SJohan Hedberg 		if (!capable(CAP_NET_ADMIN))
606bf5b30b8SZhao Hongjiang 			return -EPERM;
607b2a66aadSAntti Julku 		return hci_sock_blacklist_del(hdev, (void __user *) arg);
6080736cfa8SMarcel Holtmann 	}
609f0358568SJohan Hedberg 
610324d36edSMarcel Holtmann 	return -ENOIOCTLCMD;
6111da177e4SLinus Torvalds }
6121da177e4SLinus Torvalds 
6138fc9ced3SGustavo Padovan static int hci_sock_ioctl(struct socket *sock, unsigned int cmd,
6148fc9ced3SGustavo Padovan 			  unsigned long arg)
6151da177e4SLinus Torvalds {
6161da177e4SLinus Torvalds 	void __user *argp = (void __user *) arg;
6170736cfa8SMarcel Holtmann 	struct sock *sk = sock->sk;
6181da177e4SLinus Torvalds 	int err;
6191da177e4SLinus Torvalds 
6201da177e4SLinus Torvalds 	BT_DBG("cmd %x arg %lx", cmd, arg);
6211da177e4SLinus Torvalds 
622c1c4f956SMarcel Holtmann 	lock_sock(sk);
623c1c4f956SMarcel Holtmann 
624c1c4f956SMarcel Holtmann 	if (hci_pi(sk)->channel != HCI_CHANNEL_RAW) {
625c1c4f956SMarcel Holtmann 		err = -EBADFD;
626c1c4f956SMarcel Holtmann 		goto done;
627c1c4f956SMarcel Holtmann 	}
628c1c4f956SMarcel Holtmann 
629c1c4f956SMarcel Holtmann 	release_sock(sk);
630c1c4f956SMarcel Holtmann 
6311da177e4SLinus Torvalds 	switch (cmd) {
6321da177e4SLinus Torvalds 	case HCIGETDEVLIST:
6331da177e4SLinus Torvalds 		return hci_get_dev_list(argp);
6341da177e4SLinus Torvalds 
6351da177e4SLinus Torvalds 	case HCIGETDEVINFO:
6361da177e4SLinus Torvalds 		return hci_get_dev_info(argp);
6371da177e4SLinus Torvalds 
6381da177e4SLinus Torvalds 	case HCIGETCONNLIST:
6391da177e4SLinus Torvalds 		return hci_get_conn_list(argp);
6401da177e4SLinus Torvalds 
6411da177e4SLinus Torvalds 	case HCIDEVUP:
6421da177e4SLinus Torvalds 		if (!capable(CAP_NET_ADMIN))
643bf5b30b8SZhao Hongjiang 			return -EPERM;
6441da177e4SLinus Torvalds 		return hci_dev_open(arg);
6451da177e4SLinus Torvalds 
6461da177e4SLinus Torvalds 	case HCIDEVDOWN:
6471da177e4SLinus Torvalds 		if (!capable(CAP_NET_ADMIN))
648bf5b30b8SZhao Hongjiang 			return -EPERM;
6491da177e4SLinus Torvalds 		return hci_dev_close(arg);
6501da177e4SLinus Torvalds 
6511da177e4SLinus Torvalds 	case HCIDEVRESET:
6521da177e4SLinus Torvalds 		if (!capable(CAP_NET_ADMIN))
653bf5b30b8SZhao Hongjiang 			return -EPERM;
6541da177e4SLinus Torvalds 		return hci_dev_reset(arg);
6551da177e4SLinus Torvalds 
6561da177e4SLinus Torvalds 	case HCIDEVRESTAT:
6571da177e4SLinus Torvalds 		if (!capable(CAP_NET_ADMIN))
658bf5b30b8SZhao Hongjiang 			return -EPERM;
6591da177e4SLinus Torvalds 		return hci_dev_reset_stat(arg);
6601da177e4SLinus Torvalds 
6611da177e4SLinus Torvalds 	case HCISETSCAN:
6621da177e4SLinus Torvalds 	case HCISETAUTH:
6631da177e4SLinus Torvalds 	case HCISETENCRYPT:
6641da177e4SLinus Torvalds 	case HCISETPTYPE:
6651da177e4SLinus Torvalds 	case HCISETLINKPOL:
6661da177e4SLinus Torvalds 	case HCISETLINKMODE:
6671da177e4SLinus Torvalds 	case HCISETACLMTU:
6681da177e4SLinus Torvalds 	case HCISETSCOMTU:
6691da177e4SLinus Torvalds 		if (!capable(CAP_NET_ADMIN))
670bf5b30b8SZhao Hongjiang 			return -EPERM;
6711da177e4SLinus Torvalds 		return hci_dev_cmd(cmd, argp);
6721da177e4SLinus Torvalds 
6731da177e4SLinus Torvalds 	case HCIINQUIRY:
6741da177e4SLinus Torvalds 		return hci_inquiry(argp);
675c1c4f956SMarcel Holtmann 	}
6761da177e4SLinus Torvalds 
6771da177e4SLinus Torvalds 	lock_sock(sk);
678c1c4f956SMarcel Holtmann 
6791da177e4SLinus Torvalds 	err = hci_sock_bound_ioctl(sk, cmd, arg);
680c1c4f956SMarcel Holtmann 
681c1c4f956SMarcel Holtmann done:
6821da177e4SLinus Torvalds 	release_sock(sk);
6831da177e4SLinus Torvalds 	return err;
6841da177e4SLinus Torvalds }
6851da177e4SLinus Torvalds 
6868fc9ced3SGustavo Padovan static int hci_sock_bind(struct socket *sock, struct sockaddr *addr,
6878fc9ced3SGustavo Padovan 			 int addr_len)
6881da177e4SLinus Torvalds {
6890381101fSJohan Hedberg 	struct sockaddr_hci haddr;
6901da177e4SLinus Torvalds 	struct sock *sk = sock->sk;
6911da177e4SLinus Torvalds 	struct hci_dev *hdev = NULL;
6920381101fSJohan Hedberg 	int len, err = 0;
6931da177e4SLinus Torvalds 
6941da177e4SLinus Torvalds 	BT_DBG("sock %p sk %p", sock, sk);
6951da177e4SLinus Torvalds 
6960381101fSJohan Hedberg 	if (!addr)
6970381101fSJohan Hedberg 		return -EINVAL;
6980381101fSJohan Hedberg 
6990381101fSJohan Hedberg 	memset(&haddr, 0, sizeof(haddr));
7000381101fSJohan Hedberg 	len = min_t(unsigned int, sizeof(haddr), addr_len);
7010381101fSJohan Hedberg 	memcpy(&haddr, addr, len);
7020381101fSJohan Hedberg 
7030381101fSJohan Hedberg 	if (haddr.hci_family != AF_BLUETOOTH)
7040381101fSJohan Hedberg 		return -EINVAL;
7050381101fSJohan Hedberg 
7061da177e4SLinus Torvalds 	lock_sock(sk);
7071da177e4SLinus Torvalds 
7087cc2ade2SMarcel Holtmann 	if (sk->sk_state == BT_BOUND) {
7097cc2ade2SMarcel Holtmann 		err = -EALREADY;
7107cc2ade2SMarcel Holtmann 		goto done;
7117cc2ade2SMarcel Holtmann 	}
7127cc2ade2SMarcel Holtmann 
7137cc2ade2SMarcel Holtmann 	switch (haddr.hci_channel) {
7147cc2ade2SMarcel Holtmann 	case HCI_CHANNEL_RAW:
7157cc2ade2SMarcel Holtmann 		if (hci_pi(sk)->hdev) {
7161da177e4SLinus Torvalds 			err = -EALREADY;
7171da177e4SLinus Torvalds 			goto done;
7181da177e4SLinus Torvalds 		}
7191da177e4SLinus Torvalds 
7200381101fSJohan Hedberg 		if (haddr.hci_dev != HCI_DEV_NONE) {
7210381101fSJohan Hedberg 			hdev = hci_dev_get(haddr.hci_dev);
72270f23020SAndrei Emeltchenko 			if (!hdev) {
7231da177e4SLinus Torvalds 				err = -ENODEV;
7241da177e4SLinus Torvalds 				goto done;
7251da177e4SLinus Torvalds 			}
7261da177e4SLinus Torvalds 
7271da177e4SLinus Torvalds 			atomic_inc(&hdev->promisc);
7281da177e4SLinus Torvalds 		}
7291da177e4SLinus Torvalds 
7301da177e4SLinus Torvalds 		hci_pi(sk)->hdev = hdev;
7317cc2ade2SMarcel Holtmann 		break;
7327cc2ade2SMarcel Holtmann 
73323500189SMarcel Holtmann 	case HCI_CHANNEL_USER:
73423500189SMarcel Holtmann 		if (hci_pi(sk)->hdev) {
73523500189SMarcel Holtmann 			err = -EALREADY;
73623500189SMarcel Holtmann 			goto done;
73723500189SMarcel Holtmann 		}
73823500189SMarcel Holtmann 
73923500189SMarcel Holtmann 		if (haddr.hci_dev == HCI_DEV_NONE) {
74023500189SMarcel Holtmann 			err = -EINVAL;
74123500189SMarcel Holtmann 			goto done;
74223500189SMarcel Holtmann 		}
74323500189SMarcel Holtmann 
74410a8b86fSMarcel Holtmann 		if (!capable(CAP_NET_ADMIN)) {
74523500189SMarcel Holtmann 			err = -EPERM;
74623500189SMarcel Holtmann 			goto done;
74723500189SMarcel Holtmann 		}
74823500189SMarcel Holtmann 
74923500189SMarcel Holtmann 		hdev = hci_dev_get(haddr.hci_dev);
75023500189SMarcel Holtmann 		if (!hdev) {
75123500189SMarcel Holtmann 			err = -ENODEV;
75223500189SMarcel Holtmann 			goto done;
75323500189SMarcel Holtmann 		}
75423500189SMarcel Holtmann 
755781f899fSMarcel Holtmann 		if (test_bit(HCI_INIT, &hdev->flags) ||
756d7a5a11dSMarcel Holtmann 		    hci_dev_test_flag(hdev, HCI_SETUP) ||
757781f899fSMarcel Holtmann 		    hci_dev_test_flag(hdev, HCI_CONFIG) ||
758781f899fSMarcel Holtmann 		    (!hci_dev_test_flag(hdev, HCI_AUTO_OFF) &&
759781f899fSMarcel Holtmann 		     test_bit(HCI_UP, &hdev->flags))) {
76023500189SMarcel Holtmann 			err = -EBUSY;
76123500189SMarcel Holtmann 			hci_dev_put(hdev);
76223500189SMarcel Holtmann 			goto done;
76323500189SMarcel Holtmann 		}
76423500189SMarcel Holtmann 
765238be788SMarcel Holtmann 		if (hci_dev_test_and_set_flag(hdev, HCI_USER_CHANNEL)) {
76623500189SMarcel Holtmann 			err = -EUSERS;
76723500189SMarcel Holtmann 			hci_dev_put(hdev);
76823500189SMarcel Holtmann 			goto done;
76923500189SMarcel Holtmann 		}
77023500189SMarcel Holtmann 
77123500189SMarcel Holtmann 		mgmt_index_removed(hdev);
77223500189SMarcel Holtmann 
77323500189SMarcel Holtmann 		err = hci_dev_open(hdev->id);
77423500189SMarcel Holtmann 		if (err) {
775781f899fSMarcel Holtmann 			if (err == -EALREADY) {
776781f899fSMarcel Holtmann 				/* In case the transport is already up and
777781f899fSMarcel Holtmann 				 * running, clear the error here.
778781f899fSMarcel Holtmann 				 *
779781f899fSMarcel Holtmann 				 * This can happen when opening an user
780781f899fSMarcel Holtmann 				 * channel and HCI_AUTO_OFF grace period
781781f899fSMarcel Holtmann 				 * is still active.
782781f899fSMarcel Holtmann 				 */
783781f899fSMarcel Holtmann 				err = 0;
784781f899fSMarcel Holtmann 			} else {
785a358dc11SMarcel Holtmann 				hci_dev_clear_flag(hdev, HCI_USER_CHANNEL);
786c6521401SMarcel Holtmann 				mgmt_index_added(hdev);
78723500189SMarcel Holtmann 				hci_dev_put(hdev);
78823500189SMarcel Holtmann 				goto done;
78923500189SMarcel Holtmann 			}
790781f899fSMarcel Holtmann 		}
79123500189SMarcel Holtmann 
79223500189SMarcel Holtmann 		atomic_inc(&hdev->promisc);
79323500189SMarcel Holtmann 
79423500189SMarcel Holtmann 		hci_pi(sk)->hdev = hdev;
79523500189SMarcel Holtmann 		break;
79623500189SMarcel Holtmann 
797cd82e61cSMarcel Holtmann 	case HCI_CHANNEL_MONITOR:
798cd82e61cSMarcel Holtmann 		if (haddr.hci_dev != HCI_DEV_NONE) {
799cd82e61cSMarcel Holtmann 			err = -EINVAL;
800cd82e61cSMarcel Holtmann 			goto done;
801cd82e61cSMarcel Holtmann 		}
802cd82e61cSMarcel Holtmann 
803cd82e61cSMarcel Holtmann 		if (!capable(CAP_NET_RAW)) {
804cd82e61cSMarcel Holtmann 			err = -EPERM;
805cd82e61cSMarcel Holtmann 			goto done;
806cd82e61cSMarcel Holtmann 		}
807cd82e61cSMarcel Holtmann 
80850ebc055SMarcel Holtmann 		/* The monitor interface is restricted to CAP_NET_RAW
80950ebc055SMarcel Holtmann 		 * capabilities and with that implicitly trusted.
81050ebc055SMarcel Holtmann 		 */
81150ebc055SMarcel Holtmann 		hci_sock_set_flag(sk, HCI_SOCK_TRUSTED);
81250ebc055SMarcel Holtmann 
813cd82e61cSMarcel Holtmann 		send_monitor_replay(sk);
814cd82e61cSMarcel Holtmann 
815cd82e61cSMarcel Holtmann 		atomic_inc(&monitor_promisc);
816cd82e61cSMarcel Holtmann 		break;
817cd82e61cSMarcel Holtmann 
8187cc2ade2SMarcel Holtmann 	default:
819801c1e8dSJohan Hedberg 		if (!hci_mgmt_chan_find(haddr.hci_channel)) {
8207cc2ade2SMarcel Holtmann 			err = -EINVAL;
8217cc2ade2SMarcel Holtmann 			goto done;
8227cc2ade2SMarcel Holtmann 		}
8237cc2ade2SMarcel Holtmann 
824801c1e8dSJohan Hedberg 		if (haddr.hci_dev != HCI_DEV_NONE) {
825801c1e8dSJohan Hedberg 			err = -EINVAL;
826801c1e8dSJohan Hedberg 			goto done;
827801c1e8dSJohan Hedberg 		}
828801c1e8dSJohan Hedberg 
8291195fbb8SMarcel Holtmann 		/* Users with CAP_NET_ADMIN capabilities are allowed
8301195fbb8SMarcel Holtmann 		 * access to all management commands and events. For
8311195fbb8SMarcel Holtmann 		 * untrusted users the interface is restricted and
8321195fbb8SMarcel Holtmann 		 * also only untrusted events are sent.
83350ebc055SMarcel Holtmann 		 */
8341195fbb8SMarcel Holtmann 		if (capable(CAP_NET_ADMIN))
83550ebc055SMarcel Holtmann 			hci_sock_set_flag(sk, HCI_SOCK_TRUSTED);
83650ebc055SMarcel Holtmann 
837f9207338SMarcel Holtmann 		/* At the moment the index and unconfigured index events
838f9207338SMarcel Holtmann 		 * are enabled unconditionally. Setting them on each
839f9207338SMarcel Holtmann 		 * socket when binding keeps this functionality. They
840f9207338SMarcel Holtmann 		 * however might be cleared later and then sending of these
841f9207338SMarcel Holtmann 		 * events will be disabled, but that is then intentional.
842f6b7712eSMarcel Holtmann 		 *
843f6b7712eSMarcel Holtmann 		 * This also enables generic events that are safe to be
844f6b7712eSMarcel Holtmann 		 * received by untrusted users. Example for such events
845f6b7712eSMarcel Holtmann 		 * are changes to settings, class of device, name etc.
846f9207338SMarcel Holtmann 		 */
847f9207338SMarcel Holtmann 		if (haddr.hci_channel == HCI_CHANNEL_CONTROL) {
848f9207338SMarcel Holtmann 			hci_sock_set_flag(sk, HCI_MGMT_INDEX_EVENTS);
849f9207338SMarcel Holtmann 			hci_sock_set_flag(sk, HCI_MGMT_UNCONF_INDEX_EVENTS);
850f6b7712eSMarcel Holtmann 			hci_sock_set_flag(sk, HCI_MGMT_GENERIC_EVENTS);
851f9207338SMarcel Holtmann 		}
852801c1e8dSJohan Hedberg 		break;
853801c1e8dSJohan Hedberg 	}
854801c1e8dSJohan Hedberg 
8557cc2ade2SMarcel Holtmann 
8567cc2ade2SMarcel Holtmann 	hci_pi(sk)->channel = haddr.hci_channel;
8571da177e4SLinus Torvalds 	sk->sk_state = BT_BOUND;
8581da177e4SLinus Torvalds 
8591da177e4SLinus Torvalds done:
8601da177e4SLinus Torvalds 	release_sock(sk);
8611da177e4SLinus Torvalds 	return err;
8621da177e4SLinus Torvalds }
8631da177e4SLinus Torvalds 
8648fc9ced3SGustavo Padovan static int hci_sock_getname(struct socket *sock, struct sockaddr *addr,
8658fc9ced3SGustavo Padovan 			    int *addr_len, int peer)
8661da177e4SLinus Torvalds {
8671da177e4SLinus Torvalds 	struct sockaddr_hci *haddr = (struct sockaddr_hci *) addr;
8681da177e4SLinus Torvalds 	struct sock *sk = sock->sk;
8699d4b68b2SMarcel Holtmann 	struct hci_dev *hdev;
8709d4b68b2SMarcel Holtmann 	int err = 0;
8711da177e4SLinus Torvalds 
8721da177e4SLinus Torvalds 	BT_DBG("sock %p sk %p", sock, sk);
8731da177e4SLinus Torvalds 
87406f43cbcSMarcel Holtmann 	if (peer)
87506f43cbcSMarcel Holtmann 		return -EOPNOTSUPP;
87606f43cbcSMarcel Holtmann 
8771da177e4SLinus Torvalds 	lock_sock(sk);
8781da177e4SLinus Torvalds 
8799d4b68b2SMarcel Holtmann 	hdev = hci_pi(sk)->hdev;
8809d4b68b2SMarcel Holtmann 	if (!hdev) {
8819d4b68b2SMarcel Holtmann 		err = -EBADFD;
8829d4b68b2SMarcel Holtmann 		goto done;
8839d4b68b2SMarcel Holtmann 	}
8849d4b68b2SMarcel Holtmann 
8851da177e4SLinus Torvalds 	*addr_len = sizeof(*haddr);
8861da177e4SLinus Torvalds 	haddr->hci_family = AF_BLUETOOTH;
8877b005bd3SMarcel Holtmann 	haddr->hci_dev    = hdev->id;
8889d4b68b2SMarcel Holtmann 	haddr->hci_channel= hci_pi(sk)->channel;
8891da177e4SLinus Torvalds 
8909d4b68b2SMarcel Holtmann done:
8911da177e4SLinus Torvalds 	release_sock(sk);
8929d4b68b2SMarcel Holtmann 	return err;
8931da177e4SLinus Torvalds }
8941da177e4SLinus Torvalds 
8956039aa73SGustavo Padovan static void hci_sock_cmsg(struct sock *sk, struct msghdr *msg,
8966039aa73SGustavo Padovan 			  struct sk_buff *skb)
8971da177e4SLinus Torvalds {
8981da177e4SLinus Torvalds 	__u32 mask = hci_pi(sk)->cmsg_mask;
8991da177e4SLinus Torvalds 
9000d48d939SMarcel Holtmann 	if (mask & HCI_CMSG_DIR) {
9010d48d939SMarcel Holtmann 		int incoming = bt_cb(skb)->incoming;
9028fc9ced3SGustavo Padovan 		put_cmsg(msg, SOL_HCI, HCI_CMSG_DIR, sizeof(incoming),
9038fc9ced3SGustavo Padovan 			 &incoming);
9040d48d939SMarcel Holtmann 	}
9051da177e4SLinus Torvalds 
906a61bbcf2SPatrick McHardy 	if (mask & HCI_CMSG_TSTAMP) {
907f6e623a6SJohann Felix Soden #ifdef CONFIG_COMPAT
908f6e623a6SJohann Felix Soden 		struct compat_timeval ctv;
909f6e623a6SJohann Felix Soden #endif
910a61bbcf2SPatrick McHardy 		struct timeval tv;
911767c5eb5SMarcel Holtmann 		void *data;
912767c5eb5SMarcel Holtmann 		int len;
913a61bbcf2SPatrick McHardy 
914a61bbcf2SPatrick McHardy 		skb_get_timestamp(skb, &tv);
915767c5eb5SMarcel Holtmann 
9161da97f83SDavid S. Miller 		data = &tv;
9171da97f83SDavid S. Miller 		len = sizeof(tv);
9181da97f83SDavid S. Miller #ifdef CONFIG_COMPAT
919da88cea1SH. J. Lu 		if (!COMPAT_USE_64BIT_TIME &&
920da88cea1SH. J. Lu 		    (msg->msg_flags & MSG_CMSG_COMPAT)) {
921767c5eb5SMarcel Holtmann 			ctv.tv_sec = tv.tv_sec;
922767c5eb5SMarcel Holtmann 			ctv.tv_usec = tv.tv_usec;
923767c5eb5SMarcel Holtmann 			data = &ctv;
924767c5eb5SMarcel Holtmann 			len = sizeof(ctv);
925767c5eb5SMarcel Holtmann 		}
9261da97f83SDavid S. Miller #endif
927767c5eb5SMarcel Holtmann 
928767c5eb5SMarcel Holtmann 		put_cmsg(msg, SOL_HCI, HCI_CMSG_TSTAMP, len, data);
929a61bbcf2SPatrick McHardy 	}
9301da177e4SLinus Torvalds }
9311da177e4SLinus Torvalds 
9321b784140SYing Xue static int hci_sock_recvmsg(struct socket *sock, struct msghdr *msg, size_t len,
9331b784140SYing Xue 			    int flags)
9341da177e4SLinus Torvalds {
9351da177e4SLinus Torvalds 	int noblock = flags & MSG_DONTWAIT;
9361da177e4SLinus Torvalds 	struct sock *sk = sock->sk;
9371da177e4SLinus Torvalds 	struct sk_buff *skb;
9381da177e4SLinus Torvalds 	int copied, err;
9391da177e4SLinus Torvalds 
9401da177e4SLinus Torvalds 	BT_DBG("sock %p, sk %p", sock, sk);
9411da177e4SLinus Torvalds 
9421da177e4SLinus Torvalds 	if (flags & (MSG_OOB))
9431da177e4SLinus Torvalds 		return -EOPNOTSUPP;
9441da177e4SLinus Torvalds 
9451da177e4SLinus Torvalds 	if (sk->sk_state == BT_CLOSED)
9461da177e4SLinus Torvalds 		return 0;
9471da177e4SLinus Torvalds 
94870f23020SAndrei Emeltchenko 	skb = skb_recv_datagram(sk, flags, noblock, &err);
94970f23020SAndrei Emeltchenko 	if (!skb)
9501da177e4SLinus Torvalds 		return err;
9511da177e4SLinus Torvalds 
9521da177e4SLinus Torvalds 	copied = skb->len;
9531da177e4SLinus Torvalds 	if (len < copied) {
9541da177e4SLinus Torvalds 		msg->msg_flags |= MSG_TRUNC;
9551da177e4SLinus Torvalds 		copied = len;
9561da177e4SLinus Torvalds 	}
9571da177e4SLinus Torvalds 
958badff6d0SArnaldo Carvalho de Melo 	skb_reset_transport_header(skb);
95951f3d02bSDavid S. Miller 	err = skb_copy_datagram_msg(skb, 0, msg, copied);
9601da177e4SLinus Torvalds 
9613a208627SMarcel Holtmann 	switch (hci_pi(sk)->channel) {
9623a208627SMarcel Holtmann 	case HCI_CHANNEL_RAW:
9631da177e4SLinus Torvalds 		hci_sock_cmsg(sk, msg, skb);
9643a208627SMarcel Holtmann 		break;
96523500189SMarcel Holtmann 	case HCI_CHANNEL_USER:
966cd82e61cSMarcel Holtmann 	case HCI_CHANNEL_MONITOR:
967cd82e61cSMarcel Holtmann 		sock_recv_timestamp(msg, sk, skb);
968cd82e61cSMarcel Holtmann 		break;
969801c1e8dSJohan Hedberg 	default:
970801c1e8dSJohan Hedberg 		if (hci_mgmt_chan_find(hci_pi(sk)->channel))
971801c1e8dSJohan Hedberg 			sock_recv_timestamp(msg, sk, skb);
972801c1e8dSJohan Hedberg 		break;
9733a208627SMarcel Holtmann 	}
9741da177e4SLinus Torvalds 
9751da177e4SLinus Torvalds 	skb_free_datagram(sk, skb);
9761da177e4SLinus Torvalds 
9771da177e4SLinus Torvalds 	return err ? : copied;
9781da177e4SLinus Torvalds }
9791da177e4SLinus Torvalds 
980fa4335d7SJohan Hedberg static int hci_mgmt_cmd(struct hci_mgmt_chan *chan, struct sock *sk,
981fa4335d7SJohan Hedberg 			struct msghdr *msg, size_t msglen)
982fa4335d7SJohan Hedberg {
983fa4335d7SJohan Hedberg 	void *buf;
984fa4335d7SJohan Hedberg 	u8 *cp;
985fa4335d7SJohan Hedberg 	struct mgmt_hdr *hdr;
986fa4335d7SJohan Hedberg 	u16 opcode, index, len;
987fa4335d7SJohan Hedberg 	struct hci_dev *hdev = NULL;
988fa4335d7SJohan Hedberg 	const struct hci_mgmt_handler *handler;
989fa4335d7SJohan Hedberg 	bool var_len, no_hdev;
990fa4335d7SJohan Hedberg 	int err;
991fa4335d7SJohan Hedberg 
992fa4335d7SJohan Hedberg 	BT_DBG("got %zu bytes", msglen);
993fa4335d7SJohan Hedberg 
994fa4335d7SJohan Hedberg 	if (msglen < sizeof(*hdr))
995fa4335d7SJohan Hedberg 		return -EINVAL;
996fa4335d7SJohan Hedberg 
997fa4335d7SJohan Hedberg 	buf = kmalloc(msglen, GFP_KERNEL);
998fa4335d7SJohan Hedberg 	if (!buf)
999fa4335d7SJohan Hedberg 		return -ENOMEM;
1000fa4335d7SJohan Hedberg 
1001fa4335d7SJohan Hedberg 	if (memcpy_from_msg(buf, msg, msglen)) {
1002fa4335d7SJohan Hedberg 		err = -EFAULT;
1003fa4335d7SJohan Hedberg 		goto done;
1004fa4335d7SJohan Hedberg 	}
1005fa4335d7SJohan Hedberg 
1006fa4335d7SJohan Hedberg 	hdr = buf;
1007fa4335d7SJohan Hedberg 	opcode = __le16_to_cpu(hdr->opcode);
1008fa4335d7SJohan Hedberg 	index = __le16_to_cpu(hdr->index);
1009fa4335d7SJohan Hedberg 	len = __le16_to_cpu(hdr->len);
1010fa4335d7SJohan Hedberg 
1011fa4335d7SJohan Hedberg 	if (len != msglen - sizeof(*hdr)) {
1012fa4335d7SJohan Hedberg 		err = -EINVAL;
1013fa4335d7SJohan Hedberg 		goto done;
1014fa4335d7SJohan Hedberg 	}
1015fa4335d7SJohan Hedberg 
1016fa4335d7SJohan Hedberg 	if (opcode >= chan->handler_count ||
1017fa4335d7SJohan Hedberg 	    chan->handlers[opcode].func == NULL) {
1018fa4335d7SJohan Hedberg 		BT_DBG("Unknown op %u", opcode);
1019fa4335d7SJohan Hedberg 		err = mgmt_cmd_status(sk, index, opcode,
1020fa4335d7SJohan Hedberg 				      MGMT_STATUS_UNKNOWN_COMMAND);
1021fa4335d7SJohan Hedberg 		goto done;
1022fa4335d7SJohan Hedberg 	}
1023fa4335d7SJohan Hedberg 
1024fa4335d7SJohan Hedberg 	handler = &chan->handlers[opcode];
1025fa4335d7SJohan Hedberg 
1026fa4335d7SJohan Hedberg 	if (!hci_sock_test_flag(sk, HCI_SOCK_TRUSTED) &&
1027fa4335d7SJohan Hedberg 	    !(handler->flags & HCI_MGMT_UNTRUSTED)) {
1028fa4335d7SJohan Hedberg 		err = mgmt_cmd_status(sk, index, opcode,
1029fa4335d7SJohan Hedberg 				      MGMT_STATUS_PERMISSION_DENIED);
1030fa4335d7SJohan Hedberg 		goto done;
1031fa4335d7SJohan Hedberg 	}
1032fa4335d7SJohan Hedberg 
1033fa4335d7SJohan Hedberg 	if (index != MGMT_INDEX_NONE) {
1034fa4335d7SJohan Hedberg 		hdev = hci_dev_get(index);
1035fa4335d7SJohan Hedberg 		if (!hdev) {
1036fa4335d7SJohan Hedberg 			err = mgmt_cmd_status(sk, index, opcode,
1037fa4335d7SJohan Hedberg 					      MGMT_STATUS_INVALID_INDEX);
1038fa4335d7SJohan Hedberg 			goto done;
1039fa4335d7SJohan Hedberg 		}
1040fa4335d7SJohan Hedberg 
1041fa4335d7SJohan Hedberg 		if (hci_dev_test_flag(hdev, HCI_SETUP) ||
1042fa4335d7SJohan Hedberg 		    hci_dev_test_flag(hdev, HCI_CONFIG) ||
1043fa4335d7SJohan Hedberg 		    hci_dev_test_flag(hdev, HCI_USER_CHANNEL)) {
1044fa4335d7SJohan Hedberg 			err = mgmt_cmd_status(sk, index, opcode,
1045fa4335d7SJohan Hedberg 					      MGMT_STATUS_INVALID_INDEX);
1046fa4335d7SJohan Hedberg 			goto done;
1047fa4335d7SJohan Hedberg 		}
1048fa4335d7SJohan Hedberg 
1049fa4335d7SJohan Hedberg 		if (hci_dev_test_flag(hdev, HCI_UNCONFIGURED) &&
1050fa4335d7SJohan Hedberg 		    !(handler->flags & HCI_MGMT_UNCONFIGURED)) {
1051fa4335d7SJohan Hedberg 			err = mgmt_cmd_status(sk, index, opcode,
1052fa4335d7SJohan Hedberg 					      MGMT_STATUS_INVALID_INDEX);
1053fa4335d7SJohan Hedberg 			goto done;
1054fa4335d7SJohan Hedberg 		}
1055fa4335d7SJohan Hedberg 	}
1056fa4335d7SJohan Hedberg 
1057fa4335d7SJohan Hedberg 	no_hdev = (handler->flags & HCI_MGMT_NO_HDEV);
1058fa4335d7SJohan Hedberg 	if (no_hdev != !hdev) {
1059fa4335d7SJohan Hedberg 		err = mgmt_cmd_status(sk, index, opcode,
1060fa4335d7SJohan Hedberg 				      MGMT_STATUS_INVALID_INDEX);
1061fa4335d7SJohan Hedberg 		goto done;
1062fa4335d7SJohan Hedberg 	}
1063fa4335d7SJohan Hedberg 
1064fa4335d7SJohan Hedberg 	var_len = (handler->flags & HCI_MGMT_VAR_LEN);
1065fa4335d7SJohan Hedberg 	if ((var_len && len < handler->data_len) ||
1066fa4335d7SJohan Hedberg 	    (!var_len && len != handler->data_len)) {
1067fa4335d7SJohan Hedberg 		err = mgmt_cmd_status(sk, index, opcode,
1068fa4335d7SJohan Hedberg 				      MGMT_STATUS_INVALID_PARAMS);
1069fa4335d7SJohan Hedberg 		goto done;
1070fa4335d7SJohan Hedberg 	}
1071fa4335d7SJohan Hedberg 
1072fa4335d7SJohan Hedberg 	if (hdev && chan->hdev_init)
1073fa4335d7SJohan Hedberg 		chan->hdev_init(sk, hdev);
1074fa4335d7SJohan Hedberg 
1075fa4335d7SJohan Hedberg 	cp = buf + sizeof(*hdr);
1076fa4335d7SJohan Hedberg 
1077fa4335d7SJohan Hedberg 	err = handler->func(sk, hdev, cp, len);
1078fa4335d7SJohan Hedberg 	if (err < 0)
1079fa4335d7SJohan Hedberg 		goto done;
1080fa4335d7SJohan Hedberg 
1081fa4335d7SJohan Hedberg 	err = msglen;
1082fa4335d7SJohan Hedberg 
1083fa4335d7SJohan Hedberg done:
1084fa4335d7SJohan Hedberg 	if (hdev)
1085fa4335d7SJohan Hedberg 		hci_dev_put(hdev);
1086fa4335d7SJohan Hedberg 
1087fa4335d7SJohan Hedberg 	kfree(buf);
1088fa4335d7SJohan Hedberg 	return err;
1089fa4335d7SJohan Hedberg }
1090fa4335d7SJohan Hedberg 
10911b784140SYing Xue static int hci_sock_sendmsg(struct socket *sock, struct msghdr *msg,
10921b784140SYing Xue 			    size_t len)
10931da177e4SLinus Torvalds {
10941da177e4SLinus Torvalds 	struct sock *sk = sock->sk;
1095801c1e8dSJohan Hedberg 	struct hci_mgmt_chan *chan;
10961da177e4SLinus Torvalds 	struct hci_dev *hdev;
10971da177e4SLinus Torvalds 	struct sk_buff *skb;
10981da177e4SLinus Torvalds 	int err;
10991da177e4SLinus Torvalds 
11001da177e4SLinus Torvalds 	BT_DBG("sock %p sk %p", sock, sk);
11011da177e4SLinus Torvalds 
11021da177e4SLinus Torvalds 	if (msg->msg_flags & MSG_OOB)
11031da177e4SLinus Torvalds 		return -EOPNOTSUPP;
11041da177e4SLinus Torvalds 
11051da177e4SLinus Torvalds 	if (msg->msg_flags & ~(MSG_DONTWAIT|MSG_NOSIGNAL|MSG_ERRQUEUE))
11061da177e4SLinus Torvalds 		return -EINVAL;
11071da177e4SLinus Torvalds 
11081da177e4SLinus Torvalds 	if (len < 4 || len > HCI_MAX_FRAME_SIZE)
11091da177e4SLinus Torvalds 		return -EINVAL;
11101da177e4SLinus Torvalds 
11111da177e4SLinus Torvalds 	lock_sock(sk);
11121da177e4SLinus Torvalds 
11130381101fSJohan Hedberg 	switch (hci_pi(sk)->channel) {
11140381101fSJohan Hedberg 	case HCI_CHANNEL_RAW:
111523500189SMarcel Holtmann 	case HCI_CHANNEL_USER:
11160381101fSJohan Hedberg 		break;
1117cd82e61cSMarcel Holtmann 	case HCI_CHANNEL_MONITOR:
1118cd82e61cSMarcel Holtmann 		err = -EOPNOTSUPP;
1119cd82e61cSMarcel Holtmann 		goto done;
11200381101fSJohan Hedberg 	default:
1121801c1e8dSJohan Hedberg 		mutex_lock(&mgmt_chan_list_lock);
1122801c1e8dSJohan Hedberg 		chan = __hci_mgmt_chan_find(hci_pi(sk)->channel);
1123801c1e8dSJohan Hedberg 		if (chan)
1124fa4335d7SJohan Hedberg 			err = hci_mgmt_cmd(chan, sk, msg, len);
1125801c1e8dSJohan Hedberg 		else
11260381101fSJohan Hedberg 			err = -EINVAL;
1127801c1e8dSJohan Hedberg 
1128801c1e8dSJohan Hedberg 		mutex_unlock(&mgmt_chan_list_lock);
11290381101fSJohan Hedberg 		goto done;
11300381101fSJohan Hedberg 	}
11310381101fSJohan Hedberg 
113270f23020SAndrei Emeltchenko 	hdev = hci_pi(sk)->hdev;
113370f23020SAndrei Emeltchenko 	if (!hdev) {
11341da177e4SLinus Torvalds 		err = -EBADFD;
11351da177e4SLinus Torvalds 		goto done;
11361da177e4SLinus Torvalds 	}
11371da177e4SLinus Torvalds 
11387e21addcSMarcel Holtmann 	if (!test_bit(HCI_UP, &hdev->flags)) {
11397e21addcSMarcel Holtmann 		err = -ENETDOWN;
11407e21addcSMarcel Holtmann 		goto done;
11417e21addcSMarcel Holtmann 	}
11427e21addcSMarcel Holtmann 
114370f23020SAndrei Emeltchenko 	skb = bt_skb_send_alloc(sk, len, msg->msg_flags & MSG_DONTWAIT, &err);
114470f23020SAndrei Emeltchenko 	if (!skb)
11451da177e4SLinus Torvalds 		goto done;
11461da177e4SLinus Torvalds 
11476ce8e9ceSAl Viro 	if (memcpy_from_msg(skb_put(skb, len), msg, len)) {
11481da177e4SLinus Torvalds 		err = -EFAULT;
11491da177e4SLinus Torvalds 		goto drop;
11501da177e4SLinus Torvalds 	}
11511da177e4SLinus Torvalds 
11520d48d939SMarcel Holtmann 	bt_cb(skb)->pkt_type = *((unsigned char *) skb->data);
11531da177e4SLinus Torvalds 	skb_pull(skb, 1);
11541da177e4SLinus Torvalds 
11551bc5ad16SMarcel Holtmann 	if (hci_pi(sk)->channel == HCI_CHANNEL_USER) {
11561bc5ad16SMarcel Holtmann 		/* No permission check is needed for user channel
11571bc5ad16SMarcel Holtmann 		 * since that gets enforced when binding the socket.
11581bc5ad16SMarcel Holtmann 		 *
11591bc5ad16SMarcel Holtmann 		 * However check that the packet type is valid.
11601bc5ad16SMarcel Holtmann 		 */
11611bc5ad16SMarcel Holtmann 		if (bt_cb(skb)->pkt_type != HCI_COMMAND_PKT &&
11621bc5ad16SMarcel Holtmann 		    bt_cb(skb)->pkt_type != HCI_ACLDATA_PKT &&
11631bc5ad16SMarcel Holtmann 		    bt_cb(skb)->pkt_type != HCI_SCODATA_PKT) {
11641bc5ad16SMarcel Holtmann 			err = -EINVAL;
11651bc5ad16SMarcel Holtmann 			goto drop;
11661bc5ad16SMarcel Holtmann 		}
11671bc5ad16SMarcel Holtmann 
11681bc5ad16SMarcel Holtmann 		skb_queue_tail(&hdev->raw_q, skb);
11691bc5ad16SMarcel Holtmann 		queue_work(hdev->workqueue, &hdev->tx_work);
11701bc5ad16SMarcel Holtmann 	} else if (bt_cb(skb)->pkt_type == HCI_COMMAND_PKT) {
117183985319SHarvey Harrison 		u16 opcode = get_unaligned_le16(skb->data);
11721da177e4SLinus Torvalds 		u16 ogf = hci_opcode_ogf(opcode);
11731da177e4SLinus Torvalds 		u16 ocf = hci_opcode_ocf(opcode);
11741da177e4SLinus Torvalds 
11751da177e4SLinus Torvalds 		if (((ogf > HCI_SFLT_MAX_OGF) ||
11763bb3c755SGustavo Padovan 		     !hci_test_bit(ocf & HCI_FLT_OCF_BITS,
11773bb3c755SGustavo Padovan 				   &hci_sec_filter.ocf_mask[ogf])) &&
11781da177e4SLinus Torvalds 		    !capable(CAP_NET_RAW)) {
11791da177e4SLinus Torvalds 			err = -EPERM;
11801da177e4SLinus Torvalds 			goto drop;
11811da177e4SLinus Torvalds 		}
11821da177e4SLinus Torvalds 
1183fee746b0SMarcel Holtmann 		if (ogf == 0x3f) {
11841da177e4SLinus Torvalds 			skb_queue_tail(&hdev->raw_q, skb);
11853eff45eaSGustavo F. Padovan 			queue_work(hdev->workqueue, &hdev->tx_work);
11861da177e4SLinus Torvalds 		} else {
118749c922bbSStephen Hemminger 			/* Stand-alone HCI commands must be flagged as
118811714b3dSJohan Hedberg 			 * single-command requests.
118911714b3dSJohan Hedberg 			 */
1190db6e3e8dSJohan Hedberg 			bt_cb(skb)->req.start = true;
119111714b3dSJohan Hedberg 
11921da177e4SLinus Torvalds 			skb_queue_tail(&hdev->cmd_q, skb);
1193c347b765SGustavo F. Padovan 			queue_work(hdev->workqueue, &hdev->cmd_work);
11941da177e4SLinus Torvalds 		}
11951da177e4SLinus Torvalds 	} else {
11961da177e4SLinus Torvalds 		if (!capable(CAP_NET_RAW)) {
11971da177e4SLinus Torvalds 			err = -EPERM;
11981da177e4SLinus Torvalds 			goto drop;
11991da177e4SLinus Torvalds 		}
12001da177e4SLinus Torvalds 
12011da177e4SLinus Torvalds 		skb_queue_tail(&hdev->raw_q, skb);
12023eff45eaSGustavo F. Padovan 		queue_work(hdev->workqueue, &hdev->tx_work);
12031da177e4SLinus Torvalds 	}
12041da177e4SLinus Torvalds 
12051da177e4SLinus Torvalds 	err = len;
12061da177e4SLinus Torvalds 
12071da177e4SLinus Torvalds done:
12081da177e4SLinus Torvalds 	release_sock(sk);
12091da177e4SLinus Torvalds 	return err;
12101da177e4SLinus Torvalds 
12111da177e4SLinus Torvalds drop:
12121da177e4SLinus Torvalds 	kfree_skb(skb);
12131da177e4SLinus Torvalds 	goto done;
12141da177e4SLinus Torvalds }
12151da177e4SLinus Torvalds 
12168fc9ced3SGustavo Padovan static int hci_sock_setsockopt(struct socket *sock, int level, int optname,
12178fc9ced3SGustavo Padovan 			       char __user *optval, unsigned int len)
12181da177e4SLinus Torvalds {
12191da177e4SLinus Torvalds 	struct hci_ufilter uf = { .opcode = 0 };
12201da177e4SLinus Torvalds 	struct sock *sk = sock->sk;
12211da177e4SLinus Torvalds 	int err = 0, opt = 0;
12221da177e4SLinus Torvalds 
12231da177e4SLinus Torvalds 	BT_DBG("sk %p, opt %d", sk, optname);
12241da177e4SLinus Torvalds 
12251da177e4SLinus Torvalds 	lock_sock(sk);
12261da177e4SLinus Torvalds 
12272f39cdb7SMarcel Holtmann 	if (hci_pi(sk)->channel != HCI_CHANNEL_RAW) {
1228c2371e80SMarcel Holtmann 		err = -EBADFD;
12292f39cdb7SMarcel Holtmann 		goto done;
12302f39cdb7SMarcel Holtmann 	}
12312f39cdb7SMarcel Holtmann 
12321da177e4SLinus Torvalds 	switch (optname) {
12331da177e4SLinus Torvalds 	case HCI_DATA_DIR:
12341da177e4SLinus Torvalds 		if (get_user(opt, (int __user *)optval)) {
12351da177e4SLinus Torvalds 			err = -EFAULT;
12361da177e4SLinus Torvalds 			break;
12371da177e4SLinus Torvalds 		}
12381da177e4SLinus Torvalds 
12391da177e4SLinus Torvalds 		if (opt)
12401da177e4SLinus Torvalds 			hci_pi(sk)->cmsg_mask |= HCI_CMSG_DIR;
12411da177e4SLinus Torvalds 		else
12421da177e4SLinus Torvalds 			hci_pi(sk)->cmsg_mask &= ~HCI_CMSG_DIR;
12431da177e4SLinus Torvalds 		break;
12441da177e4SLinus Torvalds 
12451da177e4SLinus Torvalds 	case HCI_TIME_STAMP:
12461da177e4SLinus Torvalds 		if (get_user(opt, (int __user *)optval)) {
12471da177e4SLinus Torvalds 			err = -EFAULT;
12481da177e4SLinus Torvalds 			break;
12491da177e4SLinus Torvalds 		}
12501da177e4SLinus Torvalds 
12511da177e4SLinus Torvalds 		if (opt)
12521da177e4SLinus Torvalds 			hci_pi(sk)->cmsg_mask |= HCI_CMSG_TSTAMP;
12531da177e4SLinus Torvalds 		else
12541da177e4SLinus Torvalds 			hci_pi(sk)->cmsg_mask &= ~HCI_CMSG_TSTAMP;
12551da177e4SLinus Torvalds 		break;
12561da177e4SLinus Torvalds 
12571da177e4SLinus Torvalds 	case HCI_FILTER:
12580878b666SMarcel Holtmann 		{
12590878b666SMarcel Holtmann 			struct hci_filter *f = &hci_pi(sk)->filter;
12600878b666SMarcel Holtmann 
12610878b666SMarcel Holtmann 			uf.type_mask = f->type_mask;
12620878b666SMarcel Holtmann 			uf.opcode    = f->opcode;
12630878b666SMarcel Holtmann 			uf.event_mask[0] = *((u32 *) f->event_mask + 0);
12640878b666SMarcel Holtmann 			uf.event_mask[1] = *((u32 *) f->event_mask + 1);
12650878b666SMarcel Holtmann 		}
12660878b666SMarcel Holtmann 
12671da177e4SLinus Torvalds 		len = min_t(unsigned int, len, sizeof(uf));
12681da177e4SLinus Torvalds 		if (copy_from_user(&uf, optval, len)) {
12691da177e4SLinus Torvalds 			err = -EFAULT;
12701da177e4SLinus Torvalds 			break;
12711da177e4SLinus Torvalds 		}
12721da177e4SLinus Torvalds 
12731da177e4SLinus Torvalds 		if (!capable(CAP_NET_RAW)) {
12741da177e4SLinus Torvalds 			uf.type_mask &= hci_sec_filter.type_mask;
12751da177e4SLinus Torvalds 			uf.event_mask[0] &= *((u32 *) hci_sec_filter.event_mask + 0);
12761da177e4SLinus Torvalds 			uf.event_mask[1] &= *((u32 *) hci_sec_filter.event_mask + 1);
12771da177e4SLinus Torvalds 		}
12781da177e4SLinus Torvalds 
12791da177e4SLinus Torvalds 		{
12801da177e4SLinus Torvalds 			struct hci_filter *f = &hci_pi(sk)->filter;
12811da177e4SLinus Torvalds 
12821da177e4SLinus Torvalds 			f->type_mask = uf.type_mask;
12831da177e4SLinus Torvalds 			f->opcode    = uf.opcode;
12841da177e4SLinus Torvalds 			*((u32 *) f->event_mask + 0) = uf.event_mask[0];
12851da177e4SLinus Torvalds 			*((u32 *) f->event_mask + 1) = uf.event_mask[1];
12861da177e4SLinus Torvalds 		}
12871da177e4SLinus Torvalds 		break;
12881da177e4SLinus Torvalds 
12891da177e4SLinus Torvalds 	default:
12901da177e4SLinus Torvalds 		err = -ENOPROTOOPT;
12911da177e4SLinus Torvalds 		break;
12921da177e4SLinus Torvalds 	}
12931da177e4SLinus Torvalds 
12942f39cdb7SMarcel Holtmann done:
12951da177e4SLinus Torvalds 	release_sock(sk);
12961da177e4SLinus Torvalds 	return err;
12971da177e4SLinus Torvalds }
12981da177e4SLinus Torvalds 
12998fc9ced3SGustavo Padovan static int hci_sock_getsockopt(struct socket *sock, int level, int optname,
13008fc9ced3SGustavo Padovan 			       char __user *optval, int __user *optlen)
13011da177e4SLinus Torvalds {
13021da177e4SLinus Torvalds 	struct hci_ufilter uf;
13031da177e4SLinus Torvalds 	struct sock *sk = sock->sk;
1304cedc5469SMarcel Holtmann 	int len, opt, err = 0;
1305cedc5469SMarcel Holtmann 
1306cedc5469SMarcel Holtmann 	BT_DBG("sk %p, opt %d", sk, optname);
13071da177e4SLinus Torvalds 
13081da177e4SLinus Torvalds 	if (get_user(len, optlen))
13091da177e4SLinus Torvalds 		return -EFAULT;
13101da177e4SLinus Torvalds 
1311cedc5469SMarcel Holtmann 	lock_sock(sk);
1312cedc5469SMarcel Holtmann 
1313cedc5469SMarcel Holtmann 	if (hci_pi(sk)->channel != HCI_CHANNEL_RAW) {
1314c2371e80SMarcel Holtmann 		err = -EBADFD;
1315cedc5469SMarcel Holtmann 		goto done;
1316cedc5469SMarcel Holtmann 	}
1317cedc5469SMarcel Holtmann 
13181da177e4SLinus Torvalds 	switch (optname) {
13191da177e4SLinus Torvalds 	case HCI_DATA_DIR:
13201da177e4SLinus Torvalds 		if (hci_pi(sk)->cmsg_mask & HCI_CMSG_DIR)
13211da177e4SLinus Torvalds 			opt = 1;
13221da177e4SLinus Torvalds 		else
13231da177e4SLinus Torvalds 			opt = 0;
13241da177e4SLinus Torvalds 
13251da177e4SLinus Torvalds 		if (put_user(opt, optval))
1326cedc5469SMarcel Holtmann 			err = -EFAULT;
13271da177e4SLinus Torvalds 		break;
13281da177e4SLinus Torvalds 
13291da177e4SLinus Torvalds 	case HCI_TIME_STAMP:
13301da177e4SLinus Torvalds 		if (hci_pi(sk)->cmsg_mask & HCI_CMSG_TSTAMP)
13311da177e4SLinus Torvalds 			opt = 1;
13321da177e4SLinus Torvalds 		else
13331da177e4SLinus Torvalds 			opt = 0;
13341da177e4SLinus Torvalds 
13351da177e4SLinus Torvalds 		if (put_user(opt, optval))
1336cedc5469SMarcel Holtmann 			err = -EFAULT;
13371da177e4SLinus Torvalds 		break;
13381da177e4SLinus Torvalds 
13391da177e4SLinus Torvalds 	case HCI_FILTER:
13401da177e4SLinus Torvalds 		{
13411da177e4SLinus Torvalds 			struct hci_filter *f = &hci_pi(sk)->filter;
13421da177e4SLinus Torvalds 
1343e15ca9a0SMathias Krause 			memset(&uf, 0, sizeof(uf));
13441da177e4SLinus Torvalds 			uf.type_mask = f->type_mask;
13451da177e4SLinus Torvalds 			uf.opcode    = f->opcode;
13461da177e4SLinus Torvalds 			uf.event_mask[0] = *((u32 *) f->event_mask + 0);
13471da177e4SLinus Torvalds 			uf.event_mask[1] = *((u32 *) f->event_mask + 1);
13481da177e4SLinus Torvalds 		}
13491da177e4SLinus Torvalds 
13501da177e4SLinus Torvalds 		len = min_t(unsigned int, len, sizeof(uf));
13511da177e4SLinus Torvalds 		if (copy_to_user(optval, &uf, len))
1352cedc5469SMarcel Holtmann 			err = -EFAULT;
13531da177e4SLinus Torvalds 		break;
13541da177e4SLinus Torvalds 
13551da177e4SLinus Torvalds 	default:
1356cedc5469SMarcel Holtmann 		err = -ENOPROTOOPT;
13571da177e4SLinus Torvalds 		break;
13581da177e4SLinus Torvalds 	}
13591da177e4SLinus Torvalds 
1360cedc5469SMarcel Holtmann done:
1361cedc5469SMarcel Holtmann 	release_sock(sk);
1362cedc5469SMarcel Holtmann 	return err;
13631da177e4SLinus Torvalds }
13641da177e4SLinus Torvalds 
136590ddc4f0SEric Dumazet static const struct proto_ops hci_sock_ops = {
13661da177e4SLinus Torvalds 	.family		= PF_BLUETOOTH,
13671da177e4SLinus Torvalds 	.owner		= THIS_MODULE,
13681da177e4SLinus Torvalds 	.release	= hci_sock_release,
13691da177e4SLinus Torvalds 	.bind		= hci_sock_bind,
13701da177e4SLinus Torvalds 	.getname	= hci_sock_getname,
13711da177e4SLinus Torvalds 	.sendmsg	= hci_sock_sendmsg,
13721da177e4SLinus Torvalds 	.recvmsg	= hci_sock_recvmsg,
13731da177e4SLinus Torvalds 	.ioctl		= hci_sock_ioctl,
13741da177e4SLinus Torvalds 	.poll		= datagram_poll,
13751da177e4SLinus Torvalds 	.listen		= sock_no_listen,
13761da177e4SLinus Torvalds 	.shutdown	= sock_no_shutdown,
13771da177e4SLinus Torvalds 	.setsockopt	= hci_sock_setsockopt,
13781da177e4SLinus Torvalds 	.getsockopt	= hci_sock_getsockopt,
13791da177e4SLinus Torvalds 	.connect	= sock_no_connect,
13801da177e4SLinus Torvalds 	.socketpair	= sock_no_socketpair,
13811da177e4SLinus Torvalds 	.accept		= sock_no_accept,
13821da177e4SLinus Torvalds 	.mmap		= sock_no_mmap
13831da177e4SLinus Torvalds };
13841da177e4SLinus Torvalds 
13851da177e4SLinus Torvalds static struct proto hci_sk_proto = {
13861da177e4SLinus Torvalds 	.name		= "HCI",
13871da177e4SLinus Torvalds 	.owner		= THIS_MODULE,
13881da177e4SLinus Torvalds 	.obj_size	= sizeof(struct hci_pinfo)
13891da177e4SLinus Torvalds };
13901da177e4SLinus Torvalds 
13913f378b68SEric Paris static int hci_sock_create(struct net *net, struct socket *sock, int protocol,
13923f378b68SEric Paris 			   int kern)
13931da177e4SLinus Torvalds {
13941da177e4SLinus Torvalds 	struct sock *sk;
13951da177e4SLinus Torvalds 
13961da177e4SLinus Torvalds 	BT_DBG("sock %p", sock);
13971da177e4SLinus Torvalds 
13981da177e4SLinus Torvalds 	if (sock->type != SOCK_RAW)
13991da177e4SLinus Torvalds 		return -ESOCKTNOSUPPORT;
14001da177e4SLinus Torvalds 
14011da177e4SLinus Torvalds 	sock->ops = &hci_sock_ops;
14021da177e4SLinus Torvalds 
140311aa9c28SEric W. Biederman 	sk = sk_alloc(net, PF_BLUETOOTH, GFP_ATOMIC, &hci_sk_proto, kern);
14041da177e4SLinus Torvalds 	if (!sk)
14051da177e4SLinus Torvalds 		return -ENOMEM;
14061da177e4SLinus Torvalds 
14071da177e4SLinus Torvalds 	sock_init_data(sock, sk);
14081da177e4SLinus Torvalds 
14091da177e4SLinus Torvalds 	sock_reset_flag(sk, SOCK_ZAPPED);
14101da177e4SLinus Torvalds 
14111da177e4SLinus Torvalds 	sk->sk_protocol = protocol;
14121da177e4SLinus Torvalds 
14131da177e4SLinus Torvalds 	sock->state = SS_UNCONNECTED;
14141da177e4SLinus Torvalds 	sk->sk_state = BT_OPEN;
14151da177e4SLinus Torvalds 
14161da177e4SLinus Torvalds 	bt_sock_link(&hci_sk_list, sk);
14171da177e4SLinus Torvalds 	return 0;
14181da177e4SLinus Torvalds }
14191da177e4SLinus Torvalds 
1420ec1b4cf7SStephen Hemminger static const struct net_proto_family hci_sock_family_ops = {
14211da177e4SLinus Torvalds 	.family	= PF_BLUETOOTH,
14221da177e4SLinus Torvalds 	.owner	= THIS_MODULE,
14231da177e4SLinus Torvalds 	.create	= hci_sock_create,
14241da177e4SLinus Torvalds };
14251da177e4SLinus Torvalds 
14261da177e4SLinus Torvalds int __init hci_sock_init(void)
14271da177e4SLinus Torvalds {
14281da177e4SLinus Torvalds 	int err;
14291da177e4SLinus Torvalds 
1430b0a8e282SMarcel Holtmann 	BUILD_BUG_ON(sizeof(struct sockaddr_hci) > sizeof(struct sockaddr));
1431b0a8e282SMarcel Holtmann 
14321da177e4SLinus Torvalds 	err = proto_register(&hci_sk_proto, 0);
14331da177e4SLinus Torvalds 	if (err < 0)
14341da177e4SLinus Torvalds 		return err;
14351da177e4SLinus Torvalds 
14361da177e4SLinus Torvalds 	err = bt_sock_register(BTPROTO_HCI, &hci_sock_family_ops);
1437f7c86637SMasatake YAMATO 	if (err < 0) {
1438f7c86637SMasatake YAMATO 		BT_ERR("HCI socket registration failed");
14391da177e4SLinus Torvalds 		goto error;
1440f7c86637SMasatake YAMATO 	}
1441f7c86637SMasatake YAMATO 
1442b0316615SAl Viro 	err = bt_procfs_init(&init_net, "hci", &hci_sk_list, NULL);
1443f7c86637SMasatake YAMATO 	if (err < 0) {
1444f7c86637SMasatake YAMATO 		BT_ERR("Failed to create HCI proc file");
1445f7c86637SMasatake YAMATO 		bt_sock_unregister(BTPROTO_HCI);
1446f7c86637SMasatake YAMATO 		goto error;
1447f7c86637SMasatake YAMATO 	}
14481da177e4SLinus Torvalds 
14491da177e4SLinus Torvalds 	BT_INFO("HCI socket layer initialized");
14501da177e4SLinus Torvalds 
14511da177e4SLinus Torvalds 	return 0;
14521da177e4SLinus Torvalds 
14531da177e4SLinus Torvalds error:
14541da177e4SLinus Torvalds 	proto_unregister(&hci_sk_proto);
14551da177e4SLinus Torvalds 	return err;
14561da177e4SLinus Torvalds }
14571da177e4SLinus Torvalds 
1458b7440a14SAnand Gadiyar void hci_sock_cleanup(void)
14591da177e4SLinus Torvalds {
1460f7c86637SMasatake YAMATO 	bt_procfs_cleanup(&init_net, "hci");
14615e9d7f86SDavid Herrmann 	bt_sock_unregister(BTPROTO_HCI);
14621da177e4SLinus Torvalds 	proto_unregister(&hci_sk_proto);
14631da177e4SLinus Torvalds }
1464