xref: /freebsd/sys/netgraph/bluetooth/hci/ng_hci_evnt.c (revision 4ae0fa8a2f8bff9a5e96dbfd862b03172c88e697)
1878ed226SJulian Elischer /*
2878ed226SJulian Elischer  * ng_hci_evnt.c
3c398230bSWarner Losh  */
4c398230bSWarner Losh 
5c398230bSWarner Losh /*-
64d846d26SWarner Losh  * SPDX-License-Identifier: BSD-2-Clause
7fe267a55SPedro F. Giffuni  *
8878ed226SJulian Elischer  * Copyright (c) Maksim Yevmenkin <m_evmenkin@yahoo.com>
9878ed226SJulian Elischer  * All rights reserved.
10878ed226SJulian Elischer  *
11878ed226SJulian Elischer  * Redistribution and use in source and binary forms, with or without
12878ed226SJulian Elischer  * modification, are permitted provided that the following conditions
13878ed226SJulian Elischer  * are met:
14878ed226SJulian Elischer  * 1. Redistributions of source code must retain the above copyright
15878ed226SJulian Elischer  *    notice, this list of conditions and the following disclaimer.
16878ed226SJulian Elischer  * 2. Redistributions in binary form must reproduce the above copyright
17878ed226SJulian Elischer  *    notice, this list of conditions and the following disclaimer in the
18878ed226SJulian Elischer  *    documentation and/or other materials provided with the distribution.
19878ed226SJulian Elischer  *
20878ed226SJulian Elischer  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
21878ed226SJulian Elischer  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
22878ed226SJulian Elischer  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
23878ed226SJulian Elischer  * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
24878ed226SJulian Elischer  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
25878ed226SJulian Elischer  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
26878ed226SJulian Elischer  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
27878ed226SJulian Elischer  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
28878ed226SJulian Elischer  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
29878ed226SJulian Elischer  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
30878ed226SJulian Elischer  * SUCH DAMAGE.
31878ed226SJulian Elischer  *
320986ab12SMaksim Yevmenkin  * $Id: ng_hci_evnt.c,v 1.6 2003/09/08 18:57:51 max Exp $
33878ed226SJulian Elischer  */
34878ed226SJulian Elischer 
35878ed226SJulian Elischer #include <sys/param.h>
36878ed226SJulian Elischer #include <sys/systm.h>
37878ed226SJulian Elischer #include <sys/kernel.h>
38878ed226SJulian Elischer #include <sys/endian.h>
39878ed226SJulian Elischer #include <sys/malloc.h>
40878ed226SJulian Elischer #include <sys/mbuf.h>
41878ed226SJulian Elischer #include <sys/queue.h>
42878ed226SJulian Elischer #include <netgraph/ng_message.h>
43878ed226SJulian Elischer #include <netgraph/netgraph.h>
44b84b10f9SMaksim Yevmenkin #include <netgraph/bluetooth/include/ng_bluetooth.h>
45b84b10f9SMaksim Yevmenkin #include <netgraph/bluetooth/include/ng_hci.h>
46b84b10f9SMaksim Yevmenkin #include <netgraph/bluetooth/hci/ng_hci_var.h>
47b84b10f9SMaksim Yevmenkin #include <netgraph/bluetooth/hci/ng_hci_cmds.h>
48b84b10f9SMaksim Yevmenkin #include <netgraph/bluetooth/hci/ng_hci_evnt.h>
49b84b10f9SMaksim Yevmenkin #include <netgraph/bluetooth/hci/ng_hci_ulpi.h>
50b84b10f9SMaksim Yevmenkin #include <netgraph/bluetooth/hci/ng_hci_misc.h>
51878ed226SJulian Elischer 
52878ed226SJulian Elischer /******************************************************************************
53878ed226SJulian Elischer  ******************************************************************************
54878ed226SJulian Elischer  **                     HCI event processing module
55878ed226SJulian Elischer  ******************************************************************************
56878ed226SJulian Elischer  ******************************************************************************/
57878ed226SJulian Elischer 
58878ed226SJulian Elischer /*
59878ed226SJulian Elischer  * Event processing routines
60878ed226SJulian Elischer  */
61878ed226SJulian Elischer 
62878ed226SJulian Elischer static int inquiry_result             (ng_hci_unit_p, struct mbuf *);
63878ed226SJulian Elischer static int con_compl                  (ng_hci_unit_p, struct mbuf *);
64878ed226SJulian Elischer static int con_req                    (ng_hci_unit_p, struct mbuf *);
65878ed226SJulian Elischer static int discon_compl               (ng_hci_unit_p, struct mbuf *);
66f2bb1caeSJulian Elischer static int encryption_change          (ng_hci_unit_p, struct mbuf *);
67878ed226SJulian Elischer static int read_remote_features_compl (ng_hci_unit_p, struct mbuf *);
68878ed226SJulian Elischer static int qos_setup_compl            (ng_hci_unit_p, struct mbuf *);
69878ed226SJulian Elischer static int hardware_error             (ng_hci_unit_p, struct mbuf *);
70878ed226SJulian Elischer static int role_change                (ng_hci_unit_p, struct mbuf *);
71878ed226SJulian Elischer static int num_compl_pkts             (ng_hci_unit_p, struct mbuf *);
72878ed226SJulian Elischer static int mode_change                (ng_hci_unit_p, struct mbuf *);
73878ed226SJulian Elischer static int data_buffer_overflow       (ng_hci_unit_p, struct mbuf *);
74878ed226SJulian Elischer static int read_clock_offset_compl    (ng_hci_unit_p, struct mbuf *);
75878ed226SJulian Elischer static int qos_violation              (ng_hci_unit_p, struct mbuf *);
76878ed226SJulian Elischer static int page_scan_mode_change      (ng_hci_unit_p, struct mbuf *);
77878ed226SJulian Elischer static int page_scan_rep_mode_change  (ng_hci_unit_p, struct mbuf *);
78878ed226SJulian Elischer static int sync_con_queue             (ng_hci_unit_p, ng_hci_unit_con_p, int);
79878ed226SJulian Elischer static int send_data_packets          (ng_hci_unit_p, int, int);
80fbc48c2bSTakanori Watanabe static int le_event		      (ng_hci_unit_p, struct mbuf *);
81878ed226SJulian Elischer 
82878ed226SJulian Elischer /*
83878ed226SJulian Elischer  * Process HCI event packet
84878ed226SJulian Elischer  */
85878ed226SJulian Elischer 
86878ed226SJulian Elischer int
87878ed226SJulian Elischer ng_hci_process_event(ng_hci_unit_p unit, struct mbuf *event)
88878ed226SJulian Elischer {
89878ed226SJulian Elischer 	ng_hci_event_pkt_t	*hdr = NULL;
90878ed226SJulian Elischer 	int			 error = 0;
91878ed226SJulian Elischer 
92878ed226SJulian Elischer 	/* Get event packet header */
93878ed226SJulian Elischer 	NG_HCI_M_PULLUP(event, sizeof(*hdr));
94878ed226SJulian Elischer 	if (event == NULL)
95878ed226SJulian Elischer 		return (ENOBUFS);
96878ed226SJulian Elischer 
97878ed226SJulian Elischer 	hdr = mtod(event, ng_hci_event_pkt_t *);
98878ed226SJulian Elischer 
99878ed226SJulian Elischer 	NG_HCI_INFO(
100878ed226SJulian Elischer "%s: %s - got HCI event=%#x, length=%d\n",
101878ed226SJulian Elischer 		__func__, NG_NODE_NAME(unit->node), hdr->event, hdr->length);
102878ed226SJulian Elischer 
103878ed226SJulian Elischer 	/* Get rid of event header and process event */
104878ed226SJulian Elischer 	m_adj(event, sizeof(*hdr));
105878ed226SJulian Elischer 
106878ed226SJulian Elischer 	switch (hdr->event) {
107878ed226SJulian Elischer 	case NG_HCI_EVENT_INQUIRY_COMPL:
108878ed226SJulian Elischer 	case NG_HCI_EVENT_RETURN_LINK_KEYS:
109878ed226SJulian Elischer 	case NG_HCI_EVENT_PIN_CODE_REQ:
110878ed226SJulian Elischer 	case NG_HCI_EVENT_LINK_KEY_REQ:
111878ed226SJulian Elischer 	case NG_HCI_EVENT_LINK_KEY_NOTIFICATION:
112878ed226SJulian Elischer 	case NG_HCI_EVENT_LOOPBACK_COMMAND:
113878ed226SJulian Elischer 	case NG_HCI_EVENT_AUTH_COMPL:
114878ed226SJulian Elischer 	case NG_HCI_EVENT_CHANGE_CON_LINK_KEY_COMPL:
115878ed226SJulian Elischer 	case NG_HCI_EVENT_MASTER_LINK_KEY_COMPL:
116878ed226SJulian Elischer 	case NG_HCI_EVENT_FLUSH_OCCUR:	/* XXX Do we have to handle it? */
117878ed226SJulian Elischer 	case NG_HCI_EVENT_MAX_SLOT_CHANGE:
118878ed226SJulian Elischer 	case NG_HCI_EVENT_CON_PKT_TYPE_CHANGED:
119878ed226SJulian Elischer 	case NG_HCI_EVENT_BT_LOGO:
120878ed226SJulian Elischer 	case NG_HCI_EVENT_VENDOR:
121878ed226SJulian Elischer 	case NG_HCI_EVENT_REMOTE_NAME_REQ_COMPL:
122878ed226SJulian Elischer 	case NG_HCI_EVENT_READ_REMOTE_VER_INFO_COMPL:
123*4ae0fa8aSAndreas Kempe 	case NG_HCI_EVENT_IO_CAPABILITY_REQUEST:
124*4ae0fa8aSAndreas Kempe 	case NG_HCI_EVENT_SIMPLE_PAIRING_COMPLETE:
125878ed226SJulian Elischer 		/* These do not need post processing */
126878ed226SJulian Elischer 		NG_FREE_M(event);
127878ed226SJulian Elischer 		break;
128fbc48c2bSTakanori Watanabe 	case NG_HCI_EVENT_LE:
129fbc48c2bSTakanori Watanabe 		error = le_event(unit, event);
130fbc48c2bSTakanori Watanabe 		break;
131878ed226SJulian Elischer 
132878ed226SJulian Elischer 	case NG_HCI_EVENT_INQUIRY_RESULT:
133878ed226SJulian Elischer 		error = inquiry_result(unit, event);
134878ed226SJulian Elischer 		break;
135878ed226SJulian Elischer 
136878ed226SJulian Elischer 	case NG_HCI_EVENT_CON_COMPL:
137878ed226SJulian Elischer 		error = con_compl(unit, event);
138878ed226SJulian Elischer 		break;
139878ed226SJulian Elischer 
140878ed226SJulian Elischer 	case NG_HCI_EVENT_CON_REQ:
141878ed226SJulian Elischer 		error = con_req(unit, event);
142878ed226SJulian Elischer 		break;
143878ed226SJulian Elischer 
144878ed226SJulian Elischer 	case NG_HCI_EVENT_DISCON_COMPL:
145878ed226SJulian Elischer 		error = discon_compl(unit, event);
146878ed226SJulian Elischer 		break;
147878ed226SJulian Elischer 
148f2bb1caeSJulian Elischer 	case NG_HCI_EVENT_ENCRYPTION_CHANGE:
149f2bb1caeSJulian Elischer 		error = encryption_change(unit, event);
150f2bb1caeSJulian Elischer 		break;
151f2bb1caeSJulian Elischer 
152878ed226SJulian Elischer 	case NG_HCI_EVENT_READ_REMOTE_FEATURES_COMPL:
153878ed226SJulian Elischer 		error = read_remote_features_compl(unit, event);
154878ed226SJulian Elischer 		break;
155878ed226SJulian Elischer 
156878ed226SJulian Elischer 	case NG_HCI_EVENT_QOS_SETUP_COMPL:
157878ed226SJulian Elischer 		error = qos_setup_compl(unit, event);
158878ed226SJulian Elischer 		break;
159878ed226SJulian Elischer 
160878ed226SJulian Elischer 	case NG_HCI_EVENT_COMMAND_COMPL:
161878ed226SJulian Elischer 		error = ng_hci_process_command_complete(unit, event);
162878ed226SJulian Elischer 		break;
163878ed226SJulian Elischer 
164878ed226SJulian Elischer 	case NG_HCI_EVENT_COMMAND_STATUS:
165878ed226SJulian Elischer 		error = ng_hci_process_command_status(unit, event);
166878ed226SJulian Elischer 		break;
167878ed226SJulian Elischer 
168878ed226SJulian Elischer 	case NG_HCI_EVENT_HARDWARE_ERROR:
169878ed226SJulian Elischer 		error = hardware_error(unit, event);
170878ed226SJulian Elischer 		break;
171878ed226SJulian Elischer 
172878ed226SJulian Elischer 	case NG_HCI_EVENT_ROLE_CHANGE:
173878ed226SJulian Elischer 		error = role_change(unit, event);
174878ed226SJulian Elischer 		break;
175878ed226SJulian Elischer 
176878ed226SJulian Elischer 	case NG_HCI_EVENT_NUM_COMPL_PKTS:
177878ed226SJulian Elischer 		error = num_compl_pkts(unit, event);
178878ed226SJulian Elischer 		break;
179878ed226SJulian Elischer 
180878ed226SJulian Elischer 	case NG_HCI_EVENT_MODE_CHANGE:
181878ed226SJulian Elischer 		error = mode_change(unit, event);
182878ed226SJulian Elischer 		break;
183878ed226SJulian Elischer 
184878ed226SJulian Elischer 	case NG_HCI_EVENT_DATA_BUFFER_OVERFLOW:
185878ed226SJulian Elischer 		error = data_buffer_overflow(unit, event);
186878ed226SJulian Elischer 		break;
187878ed226SJulian Elischer 
188878ed226SJulian Elischer 	case NG_HCI_EVENT_READ_CLOCK_OFFSET_COMPL:
189878ed226SJulian Elischer 		error = read_clock_offset_compl(unit, event);
190878ed226SJulian Elischer 		break;
191878ed226SJulian Elischer 
192878ed226SJulian Elischer 	case NG_HCI_EVENT_QOS_VIOLATION:
193878ed226SJulian Elischer 		error = qos_violation(unit, event);
194878ed226SJulian Elischer 		break;
195878ed226SJulian Elischer 
196878ed226SJulian Elischer 	case NG_HCI_EVENT_PAGE_SCAN_MODE_CHANGE:
197878ed226SJulian Elischer 		error = page_scan_mode_change(unit, event);
198878ed226SJulian Elischer 		break;
199878ed226SJulian Elischer 
200878ed226SJulian Elischer 	case NG_HCI_EVENT_PAGE_SCAN_REP_MODE_CHANGE:
201878ed226SJulian Elischer 		error = page_scan_rep_mode_change(unit, event);
202878ed226SJulian Elischer 		break;
203878ed226SJulian Elischer 
204878ed226SJulian Elischer 	default:
205878ed226SJulian Elischer 		NG_FREE_M(event);
206878ed226SJulian Elischer 		error = EINVAL;
207878ed226SJulian Elischer 		break;
208878ed226SJulian Elischer 	}
209878ed226SJulian Elischer 
210878ed226SJulian Elischer 	return (error);
211878ed226SJulian Elischer } /* ng_hci_process_event */
212878ed226SJulian Elischer 
213878ed226SJulian Elischer /*
214878ed226SJulian Elischer  * Send ACL and/or SCO data to the unit driver
215878ed226SJulian Elischer  */
216878ed226SJulian Elischer 
217878ed226SJulian Elischer void
218878ed226SJulian Elischer ng_hci_send_data(ng_hci_unit_p unit)
219878ed226SJulian Elischer {
220878ed226SJulian Elischer 	int	count;
221878ed226SJulian Elischer 
222878ed226SJulian Elischer 	/* Send ACL data */
223878ed226SJulian Elischer 	NG_HCI_BUFF_ACL_AVAIL(unit->buffer, count);
224878ed226SJulian Elischer 
225878ed226SJulian Elischer 	NG_HCI_INFO(
226878ed226SJulian Elischer "%s: %s - sending ACL data packets, count=%d\n",
227878ed226SJulian Elischer 		__func__, NG_NODE_NAME(unit->node), count);
228878ed226SJulian Elischer 
229878ed226SJulian Elischer 	if (count > 0) {
230878ed226SJulian Elischer 		count = send_data_packets(unit, NG_HCI_LINK_ACL, count);
231878ed226SJulian Elischer 		NG_HCI_STAT_ACL_SENT(unit->stat, count);
232878ed226SJulian Elischer 		NG_HCI_BUFF_ACL_USE(unit->buffer, count);
233878ed226SJulian Elischer 	}
234878ed226SJulian Elischer 
235878ed226SJulian Elischer 	/* Send SCO data */
236878ed226SJulian Elischer 	NG_HCI_BUFF_SCO_AVAIL(unit->buffer, count);
237878ed226SJulian Elischer 
238878ed226SJulian Elischer 	NG_HCI_INFO(
239878ed226SJulian Elischer "%s: %s - sending SCO data packets, count=%d\n",
240878ed226SJulian Elischer 		__func__, NG_NODE_NAME(unit->node), count);
241878ed226SJulian Elischer 
242878ed226SJulian Elischer 	if (count > 0) {
243878ed226SJulian Elischer 		count = send_data_packets(unit, NG_HCI_LINK_SCO, count);
244878ed226SJulian Elischer 		NG_HCI_STAT_SCO_SENT(unit->stat, count);
245878ed226SJulian Elischer 		NG_HCI_BUFF_SCO_USE(unit->buffer, count);
246878ed226SJulian Elischer 	}
247878ed226SJulian Elischer } /* ng_hci_send_data */
248878ed226SJulian Elischer 
249878ed226SJulian Elischer /*
250878ed226SJulian Elischer  * Send data packets to the lower layer.
251878ed226SJulian Elischer  */
252878ed226SJulian Elischer 
253878ed226SJulian Elischer static int
254878ed226SJulian Elischer send_data_packets(ng_hci_unit_p unit, int link_type, int limit)
255878ed226SJulian Elischer {
256878ed226SJulian Elischer 	ng_hci_unit_con_p	con = NULL, winner = NULL;
257fbc48c2bSTakanori Watanabe 	int			reallink_type;
258878ed226SJulian Elischer 	item_p			item = NULL;
259878ed226SJulian Elischer 	int			min_pending, total_sent, sent, error, v;
260878ed226SJulian Elischer 
261878ed226SJulian Elischer 	for (total_sent = 0; limit > 0; ) {
262878ed226SJulian Elischer 		min_pending = 0x0fffffff;
263878ed226SJulian Elischer 		winner = NULL;
264878ed226SJulian Elischer 
265878ed226SJulian Elischer 		/*
266878ed226SJulian Elischer 		 * Find the connection that has has data to send
267878ed226SJulian Elischer 		 * and the smallest number of pending packets
268878ed226SJulian Elischer 		 */
269878ed226SJulian Elischer 
270878ed226SJulian Elischer 		LIST_FOREACH(con, &unit->con_list, next) {
271fbc48c2bSTakanori Watanabe 			reallink_type = (con->link_type == NG_HCI_LINK_SCO)?
272fbc48c2bSTakanori Watanabe 				NG_HCI_LINK_SCO: NG_HCI_LINK_ACL;
273fbc48c2bSTakanori Watanabe 			if (reallink_type != link_type){
274878ed226SJulian Elischer 				continue;
275fbc48c2bSTakanori Watanabe 			}
276878ed226SJulian Elischer 			if (NG_BT_ITEMQ_LEN(&con->conq) == 0)
277878ed226SJulian Elischer 				continue;
278878ed226SJulian Elischer 
279878ed226SJulian Elischer 			if (con->pending < min_pending) {
280878ed226SJulian Elischer 				winner = con;
281878ed226SJulian Elischer 				min_pending = con->pending;
282878ed226SJulian Elischer 			}
283878ed226SJulian Elischer 		}
284878ed226SJulian Elischer 
285878ed226SJulian Elischer 	        if (winner == NULL)
286878ed226SJulian Elischer 			break;
287878ed226SJulian Elischer 
288878ed226SJulian Elischer 		/*
289878ed226SJulian Elischer 		 * OK, we have a winner now send as much packets as we can
290878ed226SJulian Elischer 		 * Count the number of packets we have sent and then sync
291878ed226SJulian Elischer 		 * winner connection queue.
292878ed226SJulian Elischer 		 */
293878ed226SJulian Elischer 
294878ed226SJulian Elischer 		for (sent = 0; limit > 0; limit --, total_sent ++, sent ++) {
295878ed226SJulian Elischer 			NG_BT_ITEMQ_DEQUEUE(&winner->conq, item);
296878ed226SJulian Elischer 			if (item == NULL)
297878ed226SJulian Elischer 				break;
298878ed226SJulian Elischer 
299878ed226SJulian Elischer 			NG_HCI_INFO(
300878ed226SJulian Elischer "%s: %s - sending data packet, handle=%d, len=%d\n",
301878ed226SJulian Elischer 				__func__, NG_NODE_NAME(unit->node),
302878ed226SJulian Elischer 				winner->con_handle, NGI_M(item)->m_pkthdr.len);
303878ed226SJulian Elischer 
304878ed226SJulian Elischer 			/* Check if driver hook still there */
305878ed226SJulian Elischer 			v = (unit->drv != NULL && NG_HOOK_IS_VALID(unit->drv));
306878ed226SJulian Elischer 			if (!v || (unit->state & NG_HCI_UNIT_READY) !=
307878ed226SJulian Elischer 					NG_HCI_UNIT_READY) {
308878ed226SJulian Elischer 				NG_HCI_ERR(
309878ed226SJulian Elischer "%s: %s - could not send data. Hook \"%s\" is %svalid, state=%#x\n",
310878ed226SJulian Elischer 					__func__, NG_NODE_NAME(unit->node),
311878ed226SJulian Elischer 					NG_HCI_HOOK_DRV, ((v)? "" : "not "),
312878ed226SJulian Elischer 					unit->state);
313878ed226SJulian Elischer 
314878ed226SJulian Elischer 				NG_FREE_ITEM(item);
315878ed226SJulian Elischer 				error = ENOTCONN;
316878ed226SJulian Elischer 			} else {
317878ed226SJulian Elischer 				v = NGI_M(item)->m_pkthdr.len;
318878ed226SJulian Elischer 
319878ed226SJulian Elischer 				/* Give packet to raw hook */
320878ed226SJulian Elischer 				ng_hci_mtap(unit, NGI_M(item));
321878ed226SJulian Elischer 
322878ed226SJulian Elischer 				/* ... and forward item to the driver */
323878ed226SJulian Elischer 				NG_FWD_ITEM_HOOK(error, item, unit->drv);
324878ed226SJulian Elischer 			}
325878ed226SJulian Elischer 
326878ed226SJulian Elischer 			if (error != 0) {
327878ed226SJulian Elischer 				NG_HCI_ERR(
328878ed226SJulian Elischer "%s: %s - could not send data packet, handle=%d, error=%d\n",
329878ed226SJulian Elischer 					__func__, NG_NODE_NAME(unit->node),
330878ed226SJulian Elischer 					winner->con_handle, error);
331878ed226SJulian Elischer 				break;
332878ed226SJulian Elischer 			}
333878ed226SJulian Elischer 
334878ed226SJulian Elischer 			winner->pending ++;
335878ed226SJulian Elischer 			NG_HCI_STAT_BYTES_SENT(unit->stat, v);
336878ed226SJulian Elischer 		}
337878ed226SJulian Elischer 
338878ed226SJulian Elischer 		/*
339878ed226SJulian Elischer 		 * Sync connection queue for the winner
340878ed226SJulian Elischer 		 */
341878ed226SJulian Elischer 		sync_con_queue(unit, winner, sent);
342878ed226SJulian Elischer 	}
343878ed226SJulian Elischer 
344878ed226SJulian Elischer 	return (total_sent);
345878ed226SJulian Elischer } /* send_data_packets */
346878ed226SJulian Elischer 
347878ed226SJulian Elischer /*
348878ed226SJulian Elischer  * Send flow control messages to the upper layer
349878ed226SJulian Elischer  */
350878ed226SJulian Elischer 
351878ed226SJulian Elischer static int
352878ed226SJulian Elischer sync_con_queue(ng_hci_unit_p unit, ng_hci_unit_con_p con, int completed)
353878ed226SJulian Elischer {
354878ed226SJulian Elischer 	hook_p				 hook = NULL;
355878ed226SJulian Elischer 	struct ng_mesg			*msg = NULL;
356878ed226SJulian Elischer 	ng_hci_sync_con_queue_ep	*state = NULL;
357878ed226SJulian Elischer 	int				 error;
358878ed226SJulian Elischer 
359fbc48c2bSTakanori Watanabe 	hook = (con->link_type != NG_HCI_LINK_SCO)? unit->acl : unit->sco;
360878ed226SJulian Elischer 	if (hook == NULL || NG_HOOK_NOT_VALID(hook))
361878ed226SJulian Elischer 		return (ENOTCONN);
362878ed226SJulian Elischer 
363878ed226SJulian Elischer 	NG_MKMESSAGE(msg, NGM_HCI_COOKIE, NGM_HCI_SYNC_CON_QUEUE,
364878ed226SJulian Elischer 		sizeof(*state), M_NOWAIT);
365878ed226SJulian Elischer 	if (msg == NULL)
366878ed226SJulian Elischer 		return (ENOMEM);
367878ed226SJulian Elischer 
368878ed226SJulian Elischer 	state = (ng_hci_sync_con_queue_ep *)(msg->data);
369878ed226SJulian Elischer 	state->con_handle = con->con_handle;
370878ed226SJulian Elischer 	state->completed = completed;
371878ed226SJulian Elischer 
3724ae439a3SMaksim Yevmenkin 	NG_SEND_MSG_HOOK(error, unit->node, msg, hook, 0);
373878ed226SJulian Elischer 
374878ed226SJulian Elischer 	return (error);
375878ed226SJulian Elischer } /* sync_con_queue */
376fbc48c2bSTakanori Watanabe /* le meta event */
377fbc48c2bSTakanori Watanabe /* Inquiry result event */
378fbc48c2bSTakanori Watanabe static int
379fbc48c2bSTakanori Watanabe le_advertizing_report(ng_hci_unit_p unit, struct mbuf *event)
380fbc48c2bSTakanori Watanabe {
381fbc48c2bSTakanori Watanabe 	ng_hci_le_advertising_report_ep	*ep = NULL;
382fbc48c2bSTakanori Watanabe 	ng_hci_neighbor_p		 n = NULL;
383fbc48c2bSTakanori Watanabe 	bdaddr_t			 bdaddr;
384fbc48c2bSTakanori Watanabe 	int				 error = 0;
3853cba89e6STakanori Watanabe 	int				 num_reports = 0;
386fbc48c2bSTakanori Watanabe 	u_int8_t addr_type;
387fbc48c2bSTakanori Watanabe 
388fbc48c2bSTakanori Watanabe 	NG_HCI_M_PULLUP(event, sizeof(*ep));
389fbc48c2bSTakanori Watanabe 	if (event == NULL)
390fbc48c2bSTakanori Watanabe 		return (ENOBUFS);
391fbc48c2bSTakanori Watanabe 
392fbc48c2bSTakanori Watanabe 	ep = mtod(event, ng_hci_le_advertising_report_ep *);
3933cba89e6STakanori Watanabe 	num_reports = ep->num_reports;
394fbc48c2bSTakanori Watanabe 	m_adj(event, sizeof(*ep));
3953cba89e6STakanori Watanabe 	ep = NULL;
396fbc48c2bSTakanori Watanabe 
3973cba89e6STakanori Watanabe 	for (; num_reports > 0; num_reports --) {
3989f566782SJohn Baldwin 		/* event_type */
399fbc48c2bSTakanori Watanabe 		m_adj(event, sizeof(u_int8_t));
4009f566782SJohn Baldwin 
4019f566782SJohn Baldwin 		/* Get remote unit address */
402fbc48c2bSTakanori Watanabe 		NG_HCI_M_PULLUP(event, sizeof(u_int8_t));
40305c0c99eSEd Maste 		if (event == NULL) {
40405c0c99eSEd Maste 			error = ENOBUFS;
40505c0c99eSEd Maste 			goto out;
40605c0c99eSEd Maste 		}
407fbc48c2bSTakanori Watanabe 		addr_type = *mtod(event, u_int8_t *);
408fbc48c2bSTakanori Watanabe 		m_adj(event, sizeof(u_int8_t));
409fbc48c2bSTakanori Watanabe 
410fbc48c2bSTakanori Watanabe 		m_copydata(event, 0, sizeof(bdaddr), (caddr_t) &bdaddr);
411fbc48c2bSTakanori Watanabe 		m_adj(event, sizeof(bdaddr));
412fbc48c2bSTakanori Watanabe 
413fbc48c2bSTakanori Watanabe 		/* Lookup entry in the cache */
414fbc48c2bSTakanori Watanabe 		n = ng_hci_get_neighbor(unit, &bdaddr, (addr_type) ? NG_HCI_LINK_LE_RANDOM:NG_HCI_LINK_LE_PUBLIC);
415fbc48c2bSTakanori Watanabe 		if (n == NULL) {
416fbc48c2bSTakanori Watanabe 			/* Create new entry */
417fbc48c2bSTakanori Watanabe 			n = ng_hci_new_neighbor(unit);
418fbc48c2bSTakanori Watanabe 			if (n == NULL) {
419fbc48c2bSTakanori Watanabe 				error = ENOMEM;
420fbc48c2bSTakanori Watanabe 				break;
421fbc48c2bSTakanori Watanabe 			}
422fbc48c2bSTakanori Watanabe 			bcopy(&bdaddr, &n->bdaddr, sizeof(n->bdaddr));
423fbc48c2bSTakanori Watanabe 			n->addrtype = (addr_type)? NG_HCI_LINK_LE_RANDOM :
424fbc48c2bSTakanori Watanabe 			  NG_HCI_LINK_LE_PUBLIC;
425fbc48c2bSTakanori Watanabe 
426fbc48c2bSTakanori Watanabe 		} else
427fbc48c2bSTakanori Watanabe 			getmicrotime(&n->updated);
428fbc48c2bSTakanori Watanabe 
429fbc48c2bSTakanori Watanabe 		{
430fbc48c2bSTakanori Watanabe 			/*
431fbc48c2bSTakanori Watanabe 			 * TODO: Make these information
432fbc48c2bSTakanori Watanabe 			 * Available from userland.
433fbc48c2bSTakanori Watanabe 			 */
434fbc48c2bSTakanori Watanabe 			u_int8_t length_data;
435fbc48c2bSTakanori Watanabe 
4364aa92fe2STakanori Watanabe 			event = m_pullup(event, sizeof(u_int8_t));
4374aa92fe2STakanori Watanabe 			if(event == NULL){
4384aa92fe2STakanori Watanabe 				NG_HCI_WARN("%s: Event datasize Pullup Failed\n", __func__);
4394aa92fe2STakanori Watanabe 				goto out;
4404aa92fe2STakanori Watanabe 			}
441fbc48c2bSTakanori Watanabe 			length_data = *mtod(event, u_int8_t *);
442fbc48c2bSTakanori Watanabe 			m_adj(event, sizeof(u_int8_t));
4434aa92fe2STakanori Watanabe 			n->extinq_size = (length_data < NG_HCI_EXTINQ_MAX)?
4444aa92fe2STakanori Watanabe 				length_data : NG_HCI_EXTINQ_MAX;
4454aa92fe2STakanori Watanabe 
446fbc48c2bSTakanori Watanabe 			/*Advertizement data*/
4474aa92fe2STakanori Watanabe 			event = m_pullup(event, n->extinq_size);
4484aa92fe2STakanori Watanabe 			if(event == NULL){
4494aa92fe2STakanori Watanabe 				NG_HCI_WARN("%s: Event data pullup Failed\n", __func__);
4504aa92fe2STakanori Watanabe 				goto out;
4514aa92fe2STakanori Watanabe 			}
4524aa92fe2STakanori Watanabe 			m_copydata(event, 0, n->extinq_size, n->extinq_data);
4534aa92fe2STakanori Watanabe 			m_adj(event, n->extinq_size);
4544aa92fe2STakanori Watanabe 			event = m_pullup(event, sizeof(char ));
455fbc48c2bSTakanori Watanabe 			/*Get RSSI*/
4564aa92fe2STakanori Watanabe 			if(event == NULL){
4574aa92fe2STakanori Watanabe 				NG_HCI_WARN("%s: Event rssi pull up Failed\n", __func__);
4584aa92fe2STakanori Watanabe 
4594aa92fe2STakanori Watanabe 				goto out;
4604aa92fe2STakanori Watanabe 			}
4614aa92fe2STakanori Watanabe 			n->page_scan_mode = *mtod(event, char *);
462fbc48c2bSTakanori Watanabe 			m_adj(event, sizeof(u_int8_t));
463fbc48c2bSTakanori Watanabe 		}
464fbc48c2bSTakanori Watanabe 	}
4654aa92fe2STakanori Watanabe  out:
466fbc48c2bSTakanori Watanabe 	NG_FREE_M(event);
467fbc48c2bSTakanori Watanabe 
468fbc48c2bSTakanori Watanabe 	return (error);
469fbc48c2bSTakanori Watanabe } /* inquiry_result */
470fbc48c2bSTakanori Watanabe 
471fbc48c2bSTakanori Watanabe static int le_connection_complete(ng_hci_unit_p unit, struct mbuf *event)
472fbc48c2bSTakanori Watanabe {
473fbc48c2bSTakanori Watanabe 	int			 error = 0;
474fbc48c2bSTakanori Watanabe 
475fbc48c2bSTakanori Watanabe 	ng_hci_le_connection_complete_ep	*ep = NULL;
476fbc48c2bSTakanori Watanabe 	ng_hci_unit_con_p	 con = NULL;
477fbc48c2bSTakanori Watanabe 	int link_type;
478fbc48c2bSTakanori Watanabe 	uint8_t uclass[3] = {0,0,0};//dummy uclass
479fbc48c2bSTakanori Watanabe 
480fbc48c2bSTakanori Watanabe 	NG_HCI_M_PULLUP(event, sizeof(*ep));
481fbc48c2bSTakanori Watanabe 	if (event == NULL)
482fbc48c2bSTakanori Watanabe 		return (ENOBUFS);
483fbc48c2bSTakanori Watanabe 
484fbc48c2bSTakanori Watanabe 	ep = mtod(event, ng_hci_le_connection_complete_ep *);
485fbc48c2bSTakanori Watanabe 	link_type = (ep->address_type)? NG_HCI_LINK_LE_RANDOM :
486fbc48c2bSTakanori Watanabe 	  NG_HCI_LINK_LE_PUBLIC;
487fbc48c2bSTakanori Watanabe 	/*
488fbc48c2bSTakanori Watanabe 	 * Find the first connection descriptor that matches the following:
489fbc48c2bSTakanori Watanabe 	 *
490fbc48c2bSTakanori Watanabe 	 * 1) con->link_type == link_type
491fbc48c2bSTakanori Watanabe 	 * 2) con->state == NG_HCI_CON_W4_CONN_COMPLETE
492fbc48c2bSTakanori Watanabe 	 * 3) con->bdaddr == ep->address
493fbc48c2bSTakanori Watanabe 	 */
494fbc48c2bSTakanori Watanabe 	LIST_FOREACH(con, &unit->con_list, next)
495fbc48c2bSTakanori Watanabe 		if (con->link_type == link_type &&
496fbc48c2bSTakanori Watanabe 		    con->state == NG_HCI_CON_W4_CONN_COMPLETE &&
497fbc48c2bSTakanori Watanabe 		    bcmp(&con->bdaddr, &ep->address, sizeof(bdaddr_t)) == 0)
498fbc48c2bSTakanori Watanabe 			break;
499fbc48c2bSTakanori Watanabe 
500fbc48c2bSTakanori Watanabe 	/*
501fbc48c2bSTakanori Watanabe 	 * Two possible cases:
502fbc48c2bSTakanori Watanabe 	 *
503fbc48c2bSTakanori Watanabe 	 * 1) We have found connection descriptor. That means upper layer has
504fbc48c2bSTakanori Watanabe 	 *    requested this connection via LP_CON_REQ message. In this case
505fbc48c2bSTakanori Watanabe 	 *    connection must have timeout set. If ng_hci_con_untimeout() fails
506fbc48c2bSTakanori Watanabe 	 *    then timeout message already went into node's queue. In this case
507fbc48c2bSTakanori Watanabe 	 *    ignore Connection_Complete event and let timeout deal with it.
508fbc48c2bSTakanori Watanabe 	 *
509fbc48c2bSTakanori Watanabe 	 * 2) We do not have connection descriptor. That means upper layer
510fbc48c2bSTakanori Watanabe 	 *    nas not requested this connection , (less likely) we gave up
511fbc48c2bSTakanori Watanabe 	 *    on this connection (timeout) or as node act as slave role.
512fbc48c2bSTakanori Watanabe 	 *    The most likely scenario is that
513fbc48c2bSTakanori Watanabe 	 *    we have received LE_Create_Connection command
514fbc48c2bSTakanori Watanabe 	 *    from the RAW hook
515fbc48c2bSTakanori Watanabe 	 */
516fbc48c2bSTakanori Watanabe 
517fbc48c2bSTakanori Watanabe 	if (con == NULL) {
518fbc48c2bSTakanori Watanabe 		if (ep->status != 0)
519fbc48c2bSTakanori Watanabe 			goto out;
520fbc48c2bSTakanori Watanabe 
521fbc48c2bSTakanori Watanabe 		con = ng_hci_new_con(unit, link_type);
522fbc48c2bSTakanori Watanabe 		if (con == NULL) {
523fbc48c2bSTakanori Watanabe 			error = ENOMEM;
524fbc48c2bSTakanori Watanabe 			goto out;
525fbc48c2bSTakanori Watanabe 		}
526fbc48c2bSTakanori Watanabe 
527fbc48c2bSTakanori Watanabe 		con->state = NG_HCI_CON_W4_LP_CON_RSP;
528fbc48c2bSTakanori Watanabe 		ng_hci_con_timeout(con);
529fbc48c2bSTakanori Watanabe 
530fbc48c2bSTakanori Watanabe 		bcopy(&ep->address, &con->bdaddr, sizeof(con->bdaddr));
531fbc48c2bSTakanori Watanabe 		error = ng_hci_lp_con_ind(con, uclass);
532fbc48c2bSTakanori Watanabe 		if (error != 0) {
533fbc48c2bSTakanori Watanabe 			ng_hci_con_untimeout(con);
534fbc48c2bSTakanori Watanabe 			ng_hci_free_con(con);
53514803ec8STakanori Watanabe 			goto out;
536fbc48c2bSTakanori Watanabe 		}
537fbc48c2bSTakanori Watanabe 
538fbc48c2bSTakanori Watanabe 	} else if ((error = ng_hci_con_untimeout(con)) != 0)
539fbc48c2bSTakanori Watanabe 			goto out;
540fbc48c2bSTakanori Watanabe 
541fbc48c2bSTakanori Watanabe 	/*
542fbc48c2bSTakanori Watanabe 	 * Update connection descriptor and send notification
543fbc48c2bSTakanori Watanabe 	 * to the upper layers.
544fbc48c2bSTakanori Watanabe 	 */
545fbc48c2bSTakanori Watanabe 
546fbc48c2bSTakanori Watanabe 	con->con_handle = NG_HCI_CON_HANDLE(le16toh(ep->handle));
547fbc48c2bSTakanori Watanabe 	con->encryption_mode = NG_HCI_ENCRYPTION_MODE_NONE;
548fbc48c2bSTakanori Watanabe 
549fbc48c2bSTakanori Watanabe 	ng_hci_lp_con_cfm(con, ep->status);
550fbc48c2bSTakanori Watanabe 
551fbc48c2bSTakanori Watanabe 	/* Adjust connection state */
552fbc48c2bSTakanori Watanabe 	if (ep->status != 0)
553fbc48c2bSTakanori Watanabe 		ng_hci_free_con(con);
554fbc48c2bSTakanori Watanabe 	else {
555fbc48c2bSTakanori Watanabe 		con->state = NG_HCI_CON_OPEN;
556fbc48c2bSTakanori Watanabe 
557fbc48c2bSTakanori Watanabe 		/*
558fbc48c2bSTakanori Watanabe 		 * Change link policy for the ACL connections. Enable all
559fbc48c2bSTakanori Watanabe 		 * supported link modes. Enable Role switch as well if
560fbc48c2bSTakanori Watanabe 		 * device supports it.
561fbc48c2bSTakanori Watanabe 		 */
562fbc48c2bSTakanori Watanabe 	}
563fbc48c2bSTakanori Watanabe 
564fbc48c2bSTakanori Watanabe out:
565fbc48c2bSTakanori Watanabe 	NG_FREE_M(event);
566fbc48c2bSTakanori Watanabe 
567fbc48c2bSTakanori Watanabe 	return (error);
568fbc48c2bSTakanori Watanabe 
569fbc48c2bSTakanori Watanabe }
570fbc48c2bSTakanori Watanabe 
571fbc48c2bSTakanori Watanabe static int le_connection_update(ng_hci_unit_p unit, struct mbuf *event)
572fbc48c2bSTakanori Watanabe {
573fbc48c2bSTakanori Watanabe 	int error = 0;
574fbc48c2bSTakanori Watanabe 	/*TBD*/
575fbc48c2bSTakanori Watanabe 
576fbc48c2bSTakanori Watanabe 	NG_FREE_M(event);
577fbc48c2bSTakanori Watanabe 	return error;
578fbc48c2bSTakanori Watanabe 
579fbc48c2bSTakanori Watanabe }
580fbc48c2bSTakanori Watanabe static int
581fbc48c2bSTakanori Watanabe le_event(ng_hci_unit_p unit, struct mbuf *event)
582fbc48c2bSTakanori Watanabe {
583fbc48c2bSTakanori Watanabe 	int error = 0;
584fbc48c2bSTakanori Watanabe 	ng_hci_le_ep *lep;
585fbc48c2bSTakanori Watanabe 
586fbc48c2bSTakanori Watanabe 	NG_HCI_M_PULLUP(event, sizeof(*lep));
587fbc48c2bSTakanori Watanabe 	if(event ==NULL){
588fbc48c2bSTakanori Watanabe 		return ENOBUFS;
589fbc48c2bSTakanori Watanabe 	}
590fbc48c2bSTakanori Watanabe 	lep = mtod(event, ng_hci_le_ep *);
591fbc48c2bSTakanori Watanabe 	m_adj(event, sizeof(*lep));
592fbc48c2bSTakanori Watanabe 	switch(lep->subevent_code){
593fbc48c2bSTakanori Watanabe 	case NG_HCI_LEEV_CON_COMPL:
594fbc48c2bSTakanori Watanabe 		le_connection_complete(unit, event);
595fbc48c2bSTakanori Watanabe 		break;
596fbc48c2bSTakanori Watanabe 	case NG_HCI_LEEV_ADVREP:
597fbc48c2bSTakanori Watanabe 		le_advertizing_report(unit, event);
598fbc48c2bSTakanori Watanabe 		break;
599fbc48c2bSTakanori Watanabe 	case NG_HCI_LEEV_CON_UPDATE_COMPL:
600fbc48c2bSTakanori Watanabe 		le_connection_update(unit, event);
601fbc48c2bSTakanori Watanabe 		break;
602fbc48c2bSTakanori Watanabe 	case NG_HCI_LEEV_READ_REMOTE_FEATURES_COMPL:
603fbc48c2bSTakanori Watanabe 		//TBD
604fbc48c2bSTakanori Watanabe 	  /*FALLTHROUGH*/
605fbc48c2bSTakanori Watanabe 	case NG_HCI_LEEV_LONG_TERM_KEY_REQUEST:
606fbc48c2bSTakanori Watanabe 		//TBD
607fbc48c2bSTakanori Watanabe 	  /*FALLTHROUGH*/
608fbc48c2bSTakanori Watanabe 	default:
609fbc48c2bSTakanori Watanabe 	  	NG_FREE_M(event);
610fbc48c2bSTakanori Watanabe 	}
611fbc48c2bSTakanori Watanabe 	return error;
612fbc48c2bSTakanori Watanabe }
613878ed226SJulian Elischer 
614878ed226SJulian Elischer /* Inquiry result event */
615878ed226SJulian Elischer static int
616878ed226SJulian Elischer inquiry_result(ng_hci_unit_p unit, struct mbuf *event)
617878ed226SJulian Elischer {
618878ed226SJulian Elischer 	ng_hci_inquiry_result_ep	*ep = NULL;
619878ed226SJulian Elischer 	ng_hci_neighbor_p		 n = NULL;
620878ed226SJulian Elischer 	bdaddr_t			 bdaddr;
621878ed226SJulian Elischer 	int				 error = 0;
622878ed226SJulian Elischer 
623878ed226SJulian Elischer 	NG_HCI_M_PULLUP(event, sizeof(*ep));
624878ed226SJulian Elischer 	if (event == NULL)
625878ed226SJulian Elischer 		return (ENOBUFS);
626878ed226SJulian Elischer 
627878ed226SJulian Elischer 	ep = mtod(event, ng_hci_inquiry_result_ep *);
628878ed226SJulian Elischer 	m_adj(event, sizeof(*ep));
629878ed226SJulian Elischer 
630878ed226SJulian Elischer 	for (; ep->num_responses > 0; ep->num_responses --) {
631878ed226SJulian Elischer 		/* Get remote unit address */
632878ed226SJulian Elischer 		m_copydata(event, 0, sizeof(bdaddr), (caddr_t) &bdaddr);
633878ed226SJulian Elischer 		m_adj(event, sizeof(bdaddr));
634878ed226SJulian Elischer 
635878ed226SJulian Elischer 		/* Lookup entry in the cache */
636fbc48c2bSTakanori Watanabe 		n = ng_hci_get_neighbor(unit, &bdaddr, NG_HCI_LINK_ACL);
637878ed226SJulian Elischer 		if (n == NULL) {
638878ed226SJulian Elischer 			/* Create new entry */
639878ed226SJulian Elischer 			n = ng_hci_new_neighbor(unit);
640878ed226SJulian Elischer 			if (n == NULL) {
641878ed226SJulian Elischer 				error = ENOMEM;
642878ed226SJulian Elischer 				break;
643878ed226SJulian Elischer 			}
644878ed226SJulian Elischer 		} else
645878ed226SJulian Elischer 			getmicrotime(&n->updated);
646878ed226SJulian Elischer 
647878ed226SJulian Elischer 		bcopy(&bdaddr, &n->bdaddr, sizeof(n->bdaddr));
648fbc48c2bSTakanori Watanabe 		n->addrtype = NG_HCI_LINK_ACL;
649878ed226SJulian Elischer 
650878ed226SJulian Elischer 		/* XXX call m_pullup here? */
651878ed226SJulian Elischer 
652878ed226SJulian Elischer 		n->page_scan_rep_mode = *mtod(event, u_int8_t *);
653878ed226SJulian Elischer 		m_adj(event, sizeof(u_int8_t));
654878ed226SJulian Elischer 
655878ed226SJulian Elischer 		/* page_scan_period_mode */
656878ed226SJulian Elischer 		m_adj(event, sizeof(u_int8_t));
657878ed226SJulian Elischer 
658878ed226SJulian Elischer 		n->page_scan_mode = *mtod(event, u_int8_t *);
659878ed226SJulian Elischer 		m_adj(event, sizeof(u_int8_t));
660878ed226SJulian Elischer 
661878ed226SJulian Elischer 		/* class */
662878ed226SJulian Elischer 		m_adj(event, NG_HCI_CLASS_SIZE);
663878ed226SJulian Elischer 
664878ed226SJulian Elischer 		/* clock offset */
665878ed226SJulian Elischer 		m_copydata(event, 0, sizeof(n->clock_offset),
666878ed226SJulian Elischer 			(caddr_t) &n->clock_offset);
667878ed226SJulian Elischer 		n->clock_offset = le16toh(n->clock_offset);
668878ed226SJulian Elischer 	}
669878ed226SJulian Elischer 
670878ed226SJulian Elischer 	NG_FREE_M(event);
671878ed226SJulian Elischer 
672878ed226SJulian Elischer 	return (error);
673878ed226SJulian Elischer } /* inquiry_result */
674878ed226SJulian Elischer 
675878ed226SJulian Elischer /* Connection complete event */
676878ed226SJulian Elischer static int
677878ed226SJulian Elischer con_compl(ng_hci_unit_p unit, struct mbuf *event)
678878ed226SJulian Elischer {
679878ed226SJulian Elischer 	ng_hci_con_compl_ep	*ep = NULL;
680878ed226SJulian Elischer 	ng_hci_unit_con_p	 con = NULL;
681878ed226SJulian Elischer 	int			 error = 0;
682878ed226SJulian Elischer 
683878ed226SJulian Elischer 	NG_HCI_M_PULLUP(event, sizeof(*ep));
684878ed226SJulian Elischer 	if (event == NULL)
685878ed226SJulian Elischer 		return (ENOBUFS);
686878ed226SJulian Elischer 
687878ed226SJulian Elischer 	ep = mtod(event, ng_hci_con_compl_ep *);
688878ed226SJulian Elischer 
689878ed226SJulian Elischer 	/*
690878ed226SJulian Elischer 	 * Find the first connection descriptor that matches the following:
691878ed226SJulian Elischer 	 *
692878ed226SJulian Elischer 	 * 1) con->link_type == ep->link_type
693878ed226SJulian Elischer 	 * 2) con->state == NG_HCI_CON_W4_CONN_COMPLETE
694878ed226SJulian Elischer 	 * 3) con->bdaddr == ep->bdaddr
695878ed226SJulian Elischer 	 */
696878ed226SJulian Elischer 
697878ed226SJulian Elischer 	LIST_FOREACH(con, &unit->con_list, next)
698878ed226SJulian Elischer 		if (con->link_type == ep->link_type &&
699878ed226SJulian Elischer 		    con->state == NG_HCI_CON_W4_CONN_COMPLETE &&
700878ed226SJulian Elischer 		    bcmp(&con->bdaddr, &ep->bdaddr, sizeof(bdaddr_t)) == 0)
701878ed226SJulian Elischer 			break;
702878ed226SJulian Elischer 
703878ed226SJulian Elischer 	/*
704878ed226SJulian Elischer 	 * Two possible cases:
705878ed226SJulian Elischer 	 *
706878ed226SJulian Elischer 	 * 1) We have found connection descriptor. That means upper layer has
7070986ab12SMaksim Yevmenkin 	 *    requested this connection via LP_CON_REQ message. In this case
7080986ab12SMaksim Yevmenkin 	 *    connection must have timeout set. If ng_hci_con_untimeout() fails
7090986ab12SMaksim Yevmenkin 	 *    then timeout message already went into node's queue. In this case
7100986ab12SMaksim Yevmenkin 	 *    ignore Connection_Complete event and let timeout deal with it.
711878ed226SJulian Elischer 	 *
712878ed226SJulian Elischer 	 * 2) We do not have connection descriptor. That means upper layer
713878ed226SJulian Elischer 	 *    nas not requested this connection or (less likely) we gave up
714878ed226SJulian Elischer 	 *    on this connection (timeout). The most likely scenario is that
715878ed226SJulian Elischer 	 *    we have received Create_Connection/Add_SCO_Connection command
716878ed226SJulian Elischer 	 *    from the RAW hook
717878ed226SJulian Elischer 	 */
718878ed226SJulian Elischer 
719878ed226SJulian Elischer 	if (con == NULL) {
720878ed226SJulian Elischer 		if (ep->status != 0)
721878ed226SJulian Elischer 			goto out;
722878ed226SJulian Elischer 
723878ed226SJulian Elischer 		con = ng_hci_new_con(unit, ep->link_type);
724878ed226SJulian Elischer 		if (con == NULL) {
725878ed226SJulian Elischer 			error = ENOMEM;
726878ed226SJulian Elischer 			goto out;
727878ed226SJulian Elischer 		}
728878ed226SJulian Elischer 
729878ed226SJulian Elischer 		bcopy(&ep->bdaddr, &con->bdaddr, sizeof(con->bdaddr));
7300986ab12SMaksim Yevmenkin 	} else if ((error = ng_hci_con_untimeout(con)) != 0)
7310986ab12SMaksim Yevmenkin 			goto out;
732878ed226SJulian Elischer 
733878ed226SJulian Elischer 	/*
734878ed226SJulian Elischer 	 * Update connection descriptor and send notification
735878ed226SJulian Elischer 	 * to the upper layers.
736878ed226SJulian Elischer 	 */
737878ed226SJulian Elischer 
738878ed226SJulian Elischer 	con->con_handle = NG_HCI_CON_HANDLE(le16toh(ep->con_handle));
739878ed226SJulian Elischer 	con->encryption_mode = ep->encryption_mode;
740878ed226SJulian Elischer 
741878ed226SJulian Elischer 	ng_hci_lp_con_cfm(con, ep->status);
742878ed226SJulian Elischer 
743878ed226SJulian Elischer 	/* Adjust connection state */
744878ed226SJulian Elischer 	if (ep->status != 0)
745878ed226SJulian Elischer 		ng_hci_free_con(con);
746878ed226SJulian Elischer 	else {
747878ed226SJulian Elischer 		con->state = NG_HCI_CON_OPEN;
748878ed226SJulian Elischer 
749878ed226SJulian Elischer 		/*
750878ed226SJulian Elischer 		 * Change link policy for the ACL connections. Enable all
751878ed226SJulian Elischer 		 * supported link modes. Enable Role switch as well if
752878ed226SJulian Elischer 		 * device supports it.
753878ed226SJulian Elischer 		 */
754878ed226SJulian Elischer 
755878ed226SJulian Elischer 		if (ep->link_type == NG_HCI_LINK_ACL) {
756878ed226SJulian Elischer 			struct __link_policy {
757878ed226SJulian Elischer 				ng_hci_cmd_pkt_t			 hdr;
758878ed226SJulian Elischer 				ng_hci_write_link_policy_settings_cp	 cp;
759878ed226SJulian Elischer 			} __attribute__ ((packed))			*lp;
760878ed226SJulian Elischer 			struct mbuf					*m;
761878ed226SJulian Elischer 
762eb1b1807SGleb Smirnoff 			MGETHDR(m, M_NOWAIT, MT_DATA);
763878ed226SJulian Elischer 			if (m != NULL) {
764878ed226SJulian Elischer 				m->m_pkthdr.len = m->m_len = sizeof(*lp);
765878ed226SJulian Elischer 				lp = mtod(m, struct __link_policy *);
766878ed226SJulian Elischer 
767878ed226SJulian Elischer 				lp->hdr.type = NG_HCI_CMD_PKT;
768878ed226SJulian Elischer 				lp->hdr.opcode = htole16(NG_HCI_OPCODE(
769878ed226SJulian Elischer 					NG_HCI_OGF_LINK_POLICY,
770878ed226SJulian Elischer 					NG_HCI_OCF_WRITE_LINK_POLICY_SETTINGS));
771878ed226SJulian Elischer 				lp->hdr.length = sizeof(lp->cp);
772878ed226SJulian Elischer 
773878ed226SJulian Elischer 				lp->cp.con_handle = ep->con_handle;
774878ed226SJulian Elischer 
775878ed226SJulian Elischer 				lp->cp.settings = 0;
776f2bb1caeSJulian Elischer 				if ((unit->features[0] & NG_HCI_LMP_SWITCH) &&
777f2bb1caeSJulian Elischer 				    unit->role_switch)
778878ed226SJulian Elischer 					lp->cp.settings |= 0x1;
779878ed226SJulian Elischer 				if (unit->features[0] & NG_HCI_LMP_HOLD_MODE)
780878ed226SJulian Elischer 					lp->cp.settings |= 0x2;
781878ed226SJulian Elischer 				if (unit->features[0] & NG_HCI_LMP_SNIFF_MODE)
782878ed226SJulian Elischer 					lp->cp.settings |= 0x4;
783878ed226SJulian Elischer 				if (unit->features[1] & NG_HCI_LMP_PARK_MODE)
784878ed226SJulian Elischer 					lp->cp.settings |= 0x8;
785878ed226SJulian Elischer 
786878ed226SJulian Elischer 				lp->cp.settings &= unit->link_policy_mask;
787878ed226SJulian Elischer 				lp->cp.settings = htole16(lp->cp.settings);
788878ed226SJulian Elischer 
789878ed226SJulian Elischer 				NG_BT_MBUFQ_ENQUEUE(&unit->cmdq, m);
790878ed226SJulian Elischer 				if (!(unit->state & NG_HCI_UNIT_COMMAND_PENDING))
791878ed226SJulian Elischer 					ng_hci_send_command(unit);
792878ed226SJulian Elischer 			}
793878ed226SJulian Elischer 		}
794878ed226SJulian Elischer 	}
795878ed226SJulian Elischer out:
796878ed226SJulian Elischer 	NG_FREE_M(event);
797878ed226SJulian Elischer 
798878ed226SJulian Elischer 	return (error);
7990986ab12SMaksim Yevmenkin } /* con_compl */
800878ed226SJulian Elischer 
801878ed226SJulian Elischer /* Connection request event */
802878ed226SJulian Elischer static int
803878ed226SJulian Elischer con_req(ng_hci_unit_p unit, struct mbuf *event)
804878ed226SJulian Elischer {
805878ed226SJulian Elischer 	ng_hci_con_req_ep	*ep = NULL;
806878ed226SJulian Elischer 	ng_hci_unit_con_p	 con = NULL;
807878ed226SJulian Elischer 	int			 error = 0;
808878ed226SJulian Elischer 
809878ed226SJulian Elischer 	NG_HCI_M_PULLUP(event, sizeof(*ep));
810878ed226SJulian Elischer 	if (event == NULL)
811878ed226SJulian Elischer 		return (ENOBUFS);
812878ed226SJulian Elischer 
813878ed226SJulian Elischer 	ep = mtod(event, ng_hci_con_req_ep *);
814878ed226SJulian Elischer 
815878ed226SJulian Elischer 	/*
816878ed226SJulian Elischer 	 * Find the first connection descriptor that matches the following:
817878ed226SJulian Elischer 	 *
818878ed226SJulian Elischer 	 * 1) con->link_type == ep->link_type
819878ed226SJulian Elischer 	 *
820878ed226SJulian Elischer 	 * 2) con->state == NG_HCI_CON_W4_LP_CON_RSP ||
821878ed226SJulian Elischer 	 *    con->state == NG_HCI_CON_W4_CONN_COMPL
822878ed226SJulian Elischer 	 *
823878ed226SJulian Elischer 	 * 3) con->bdaddr == ep->bdaddr
824878ed226SJulian Elischer 	 *
825878ed226SJulian Elischer 	 * Possible cases:
826878ed226SJulian Elischer 	 *
827878ed226SJulian Elischer 	 * 1) We do not have connection descriptor. This is simple. Create
828878ed226SJulian Elischer 	 *    new fresh connection descriptor and send notification to the
829878ed226SJulian Elischer 	 *    appropriate upstream hook (based on link_type).
830878ed226SJulian Elischer 	 *
831878ed226SJulian Elischer 	 * 2) We found connection handle. This is more complicated.
832878ed226SJulian Elischer 	 *
833878ed226SJulian Elischer 	 * 2.1) ACL links
834878ed226SJulian Elischer 	 *
835878ed226SJulian Elischer 	 *      Since only one ACL link can exist between each pair of
836878ed226SJulian Elischer 	 *      units then we have a race. Our upper layer has requested
837878ed226SJulian Elischer 	 *      an ACL connection to the remote unit, but we did not send
838878ed226SJulian Elischer 	 *      command yet. At the same time the remote unit has requested
839878ed226SJulian Elischer 	 *      an ACL connection from us. In this case we will ignore
840878ed226SJulian Elischer 	 *	Connection_Request event. This probably will cause connect
841878ed226SJulian Elischer 	 *      failure	on both units.
842878ed226SJulian Elischer 	 *
843878ed226SJulian Elischer 	 * 2.2) SCO links
844878ed226SJulian Elischer 	 *
845878ed226SJulian Elischer 	 *      The spec on page 45 says :
846878ed226SJulian Elischer 	 *
847878ed226SJulian Elischer 	 *      "The master can support up to three SCO links to the same
848878ed226SJulian Elischer 	 *       slave or to different slaves. A slave can support up to
849878ed226SJulian Elischer 	 *       three SCO links from the same master, or two SCO links if
850878ed226SJulian Elischer 	 *       the links originate from different masters."
851878ed226SJulian Elischer 	 *
852878ed226SJulian Elischer 	 *      The only problem is how to handle multiple SCO links between
853878ed226SJulian Elischer 	 *      matster and slave. For now we will assume that multiple SCO
854878ed226SJulian Elischer 	 *      links MUST be opened one after another.
855878ed226SJulian Elischer 	 */
856878ed226SJulian Elischer 
857878ed226SJulian Elischer 	LIST_FOREACH(con, &unit->con_list, next)
858878ed226SJulian Elischer 		if (con->link_type == ep->link_type &&
859878ed226SJulian Elischer 		    (con->state == NG_HCI_CON_W4_LP_CON_RSP ||
860878ed226SJulian Elischer 		     con->state == NG_HCI_CON_W4_CONN_COMPLETE) &&
861878ed226SJulian Elischer 		    bcmp(&con->bdaddr, &ep->bdaddr, sizeof(bdaddr_t)) == 0)
862878ed226SJulian Elischer 			break;
863878ed226SJulian Elischer 
864878ed226SJulian Elischer 	if (con == NULL) {
865878ed226SJulian Elischer 		con = ng_hci_new_con(unit, ep->link_type);
866878ed226SJulian Elischer 		if (con != NULL) {
867878ed226SJulian Elischer 			bcopy(&ep->bdaddr, &con->bdaddr, sizeof(con->bdaddr));
868878ed226SJulian Elischer 
869878ed226SJulian Elischer 			con->state = NG_HCI_CON_W4_LP_CON_RSP;
870878ed226SJulian Elischer 			ng_hci_con_timeout(con);
871878ed226SJulian Elischer 
872878ed226SJulian Elischer 			error = ng_hci_lp_con_ind(con, ep->uclass);
8730986ab12SMaksim Yevmenkin 			if (error != 0) {
8740986ab12SMaksim Yevmenkin 				ng_hci_con_untimeout(con);
875878ed226SJulian Elischer 				ng_hci_free_con(con);
8760986ab12SMaksim Yevmenkin 			}
877878ed226SJulian Elischer 		} else
878878ed226SJulian Elischer 			error = ENOMEM;
879878ed226SJulian Elischer 	}
880878ed226SJulian Elischer 
881878ed226SJulian Elischer 	NG_FREE_M(event);
882878ed226SJulian Elischer 
883878ed226SJulian Elischer 	return (error);
884878ed226SJulian Elischer } /* con_req */
885878ed226SJulian Elischer 
886878ed226SJulian Elischer /* Disconnect complete event */
887878ed226SJulian Elischer static int
888878ed226SJulian Elischer discon_compl(ng_hci_unit_p unit, struct mbuf *event)
889878ed226SJulian Elischer {
890878ed226SJulian Elischer 	ng_hci_discon_compl_ep	*ep = NULL;
891878ed226SJulian Elischer 	ng_hci_unit_con_p	 con = NULL;
892878ed226SJulian Elischer 	int			 error = 0;
893878ed226SJulian Elischer 	u_int16_t		 h;
894878ed226SJulian Elischer 
895878ed226SJulian Elischer 	NG_HCI_M_PULLUP(event, sizeof(*ep));
896878ed226SJulian Elischer 	if (event == NULL)
897878ed226SJulian Elischer 		return (ENOBUFS);
898878ed226SJulian Elischer 
899878ed226SJulian Elischer 	ep = mtod(event, ng_hci_discon_compl_ep *);
900878ed226SJulian Elischer 
901878ed226SJulian Elischer 	/*
902878ed226SJulian Elischer 	 * XXX
903878ed226SJulian Elischer 	 * Do we have to send notification if ep->status != 0?
904878ed226SJulian Elischer 	 * For now we will send notification for both ACL and SCO connections
905878ed226SJulian Elischer 	 * ONLY if ep->status == 0.
906878ed226SJulian Elischer 	 */
907878ed226SJulian Elischer 
908878ed226SJulian Elischer 	if (ep->status == 0) {
909878ed226SJulian Elischer 		h = NG_HCI_CON_HANDLE(le16toh(ep->con_handle));
910878ed226SJulian Elischer 		con = ng_hci_con_by_handle(unit, h);
911878ed226SJulian Elischer 		if (con != NULL) {
912878ed226SJulian Elischer 			error = ng_hci_lp_discon_ind(con, ep->reason);
9130986ab12SMaksim Yevmenkin 
9140986ab12SMaksim Yevmenkin 			/* Remove all timeouts (if any) */
9150986ab12SMaksim Yevmenkin 			if (con->flags & NG_HCI_CON_TIMEOUT_PENDING)
9160986ab12SMaksim Yevmenkin 				ng_hci_con_untimeout(con);
9170986ab12SMaksim Yevmenkin 
918878ed226SJulian Elischer 			ng_hci_free_con(con);
919878ed226SJulian Elischer 		} else {
920878ed226SJulian Elischer 			NG_HCI_ALERT(
921878ed226SJulian Elischer "%s: %s - invalid connection handle=%d\n",
922878ed226SJulian Elischer 				__func__, NG_NODE_NAME(unit->node), h);
923878ed226SJulian Elischer 			error = ENOENT;
924878ed226SJulian Elischer 		}
925878ed226SJulian Elischer 	}
926878ed226SJulian Elischer 
927878ed226SJulian Elischer 	NG_FREE_M(event);
928878ed226SJulian Elischer 
929878ed226SJulian Elischer 	return (error);
930878ed226SJulian Elischer } /* discon_compl */
931878ed226SJulian Elischer 
932f2bb1caeSJulian Elischer /* Encryption change event */
933f2bb1caeSJulian Elischer static int
934f2bb1caeSJulian Elischer encryption_change(ng_hci_unit_p unit, struct mbuf *event)
935f2bb1caeSJulian Elischer {
936f2bb1caeSJulian Elischer 	ng_hci_encryption_change_ep	*ep = NULL;
937f2bb1caeSJulian Elischer 	ng_hci_unit_con_p		 con = NULL;
938f2bb1caeSJulian Elischer 	int				 error = 0;
939483ed395STakanori Watanabe 	u_int16_t	h;
940f2bb1caeSJulian Elischer 
941f2bb1caeSJulian Elischer 	NG_HCI_M_PULLUP(event, sizeof(*ep));
942f2bb1caeSJulian Elischer 	if (event == NULL)
943f2bb1caeSJulian Elischer 		return (ENOBUFS);
944f2bb1caeSJulian Elischer 
945f2bb1caeSJulian Elischer 	ep = mtod(event, ng_hci_encryption_change_ep *);
946483ed395STakanori Watanabe 	h = NG_HCI_CON_HANDLE(le16toh(ep->con_handle));
947483ed395STakanori Watanabe 	con = ng_hci_con_by_handle(unit, h);
948f2bb1caeSJulian Elischer 
949f2bb1caeSJulian Elischer 	if (ep->status == 0) {
950f2bb1caeSJulian Elischer 		if (con == NULL) {
951f2bb1caeSJulian Elischer 			NG_HCI_ALERT(
952f2bb1caeSJulian Elischer "%s: %s - invalid connection handle=%d\n",
953f2bb1caeSJulian Elischer 				__func__, NG_NODE_NAME(unit->node), h);
954f2bb1caeSJulian Elischer 			error = ENOENT;
9553a601a23STakanori Watanabe 		} else if (con->link_type == NG_HCI_LINK_SCO) {
956f2bb1caeSJulian Elischer 			NG_HCI_ALERT(
957f2bb1caeSJulian Elischer "%s: %s - invalid link type=%d\n",
958f2bb1caeSJulian Elischer 				__func__, NG_NODE_NAME(unit->node),
959f2bb1caeSJulian Elischer 				con->link_type);
960f2bb1caeSJulian Elischer 			error = EINVAL;
961f2bb1caeSJulian Elischer 		} else if (ep->encryption_enable)
962f2bb1caeSJulian Elischer 			/* XXX is that true? */
963f2bb1caeSJulian Elischer 			con->encryption_mode = NG_HCI_ENCRYPTION_MODE_P2P;
964f2bb1caeSJulian Elischer 		else
965f2bb1caeSJulian Elischer 			con->encryption_mode = NG_HCI_ENCRYPTION_MODE_NONE;
966f2bb1caeSJulian Elischer 	} else
967f2bb1caeSJulian Elischer 		NG_HCI_ERR(
968f2bb1caeSJulian Elischer "%s: %s - failed to change encryption mode, status=%d\n",
969f2bb1caeSJulian Elischer 			__func__, NG_NODE_NAME(unit->node), ep->status);
970f2bb1caeSJulian Elischer 
971483ed395STakanori Watanabe 	/*Anyway, propagete encryption status to upper layer*/
972483ed395STakanori Watanabe 	ng_hci_lp_enc_change(con, con->encryption_mode);
973483ed395STakanori Watanabe 
974f2bb1caeSJulian Elischer 	NG_FREE_M(event);
975f2bb1caeSJulian Elischer 
976f2bb1caeSJulian Elischer 	return (error);
977f2bb1caeSJulian Elischer } /* encryption_change */
978f2bb1caeSJulian Elischer 
979878ed226SJulian Elischer /* Read remote feature complete event */
980878ed226SJulian Elischer static int
981878ed226SJulian Elischer read_remote_features_compl(ng_hci_unit_p unit, struct mbuf *event)
982878ed226SJulian Elischer {
983878ed226SJulian Elischer 	ng_hci_read_remote_features_compl_ep	*ep = NULL;
984878ed226SJulian Elischer 	ng_hci_unit_con_p			 con = NULL;
985878ed226SJulian Elischer 	ng_hci_neighbor_p			 n = NULL;
986878ed226SJulian Elischer 	u_int16_t				 h;
987878ed226SJulian Elischer 	int					 error = 0;
988878ed226SJulian Elischer 
989878ed226SJulian Elischer 	NG_HCI_M_PULLUP(event, sizeof(*ep));
990878ed226SJulian Elischer 	if (event == NULL)
991878ed226SJulian Elischer 		return (ENOBUFS);
992878ed226SJulian Elischer 
993878ed226SJulian Elischer 	ep = mtod(event, ng_hci_read_remote_features_compl_ep *);
994878ed226SJulian Elischer 
995878ed226SJulian Elischer 	if (ep->status == 0) {
996878ed226SJulian Elischer 		/* Check if we have this connection handle */
997878ed226SJulian Elischer 		h = NG_HCI_CON_HANDLE(le16toh(ep->con_handle));
998878ed226SJulian Elischer 		con = ng_hci_con_by_handle(unit, h);
999878ed226SJulian Elischer 		if (con == NULL) {
1000878ed226SJulian Elischer 			NG_HCI_ALERT(
1001878ed226SJulian Elischer "%s: %s - invalid connection handle=%d\n",
1002878ed226SJulian Elischer 				__func__, NG_NODE_NAME(unit->node), h);
1003878ed226SJulian Elischer 			error = ENOENT;
1004878ed226SJulian Elischer 			goto out;
1005878ed226SJulian Elischer 		}
1006878ed226SJulian Elischer 
1007878ed226SJulian Elischer 		/* Update cache entry */
1008fbc48c2bSTakanori Watanabe 		n = ng_hci_get_neighbor(unit, &con->bdaddr, NG_HCI_LINK_ACL);
1009878ed226SJulian Elischer 		if (n == NULL) {
1010878ed226SJulian Elischer 			n = ng_hci_new_neighbor(unit);
1011878ed226SJulian Elischer 			if (n == NULL) {
1012878ed226SJulian Elischer 				error = ENOMEM;
1013878ed226SJulian Elischer 				goto out;
1014878ed226SJulian Elischer 			}
1015878ed226SJulian Elischer 
1016878ed226SJulian Elischer 			bcopy(&con->bdaddr, &n->bdaddr, sizeof(n->bdaddr));
1017fbc48c2bSTakanori Watanabe 			n->addrtype = NG_HCI_LINK_ACL;
1018878ed226SJulian Elischer 		} else
1019878ed226SJulian Elischer 			getmicrotime(&n->updated);
1020878ed226SJulian Elischer 
1021878ed226SJulian Elischer 		bcopy(ep->features, n->features, sizeof(n->features));
1022878ed226SJulian Elischer 	} else
1023878ed226SJulian Elischer 		NG_HCI_ERR(
1024878ed226SJulian Elischer "%s: %s - failed to read remote unit features, status=%d\n",
1025878ed226SJulian Elischer 			__func__, NG_NODE_NAME(unit->node), ep->status);
1026878ed226SJulian Elischer out:
1027878ed226SJulian Elischer 	NG_FREE_M(event);
1028878ed226SJulian Elischer 
1029878ed226SJulian Elischer 	return (error);
1030878ed226SJulian Elischer } /* read_remote_features_compl */
1031878ed226SJulian Elischer 
1032878ed226SJulian Elischer /* QoS setup complete event */
1033878ed226SJulian Elischer static int
1034878ed226SJulian Elischer qos_setup_compl(ng_hci_unit_p unit, struct mbuf *event)
1035878ed226SJulian Elischer {
1036878ed226SJulian Elischer 	ng_hci_qos_setup_compl_ep	*ep = NULL;
1037878ed226SJulian Elischer 	ng_hci_unit_con_p		 con = NULL;
1038878ed226SJulian Elischer 	u_int16_t			 h;
1039878ed226SJulian Elischer 	int				 error = 0;
1040878ed226SJulian Elischer 
1041878ed226SJulian Elischer 	NG_HCI_M_PULLUP(event, sizeof(*ep));
1042878ed226SJulian Elischer 	if (event == NULL)
1043878ed226SJulian Elischer 		return (ENOBUFS);
1044878ed226SJulian Elischer 
1045878ed226SJulian Elischer 	ep = mtod(event, ng_hci_qos_setup_compl_ep *);
1046878ed226SJulian Elischer 
1047878ed226SJulian Elischer 	/* Check if we have this connection handle */
1048878ed226SJulian Elischer 	h = NG_HCI_CON_HANDLE(le16toh(ep->con_handle));
1049878ed226SJulian Elischer 	con = ng_hci_con_by_handle(unit, h);
1050878ed226SJulian Elischer 	if (con == NULL) {
1051878ed226SJulian Elischer 		NG_HCI_ALERT(
1052878ed226SJulian Elischer "%s: %s - invalid connection handle=%d\n",
1053878ed226SJulian Elischer 			__func__, NG_NODE_NAME(unit->node), h);
1054878ed226SJulian Elischer 		error = ENOENT;
1055878ed226SJulian Elischer 	} else if (con->link_type != NG_HCI_LINK_ACL) {
1056878ed226SJulian Elischer 		NG_HCI_ALERT(
1057878ed226SJulian Elischer "%s: %s - invalid link type=%d, handle=%d\n",
1058878ed226SJulian Elischer 			__func__, NG_NODE_NAME(unit->node), con->link_type, h);
1059878ed226SJulian Elischer 		error = EINVAL;
1060878ed226SJulian Elischer 	} else if (con->state != NG_HCI_CON_OPEN) {
1061878ed226SJulian Elischer 		NG_HCI_ALERT(
1062878ed226SJulian Elischer "%s: %s - invalid connection state=%d, handle=%d\n",
1063878ed226SJulian Elischer 			__func__, NG_NODE_NAME(unit->node),
1064878ed226SJulian Elischer 			con->state, h);
1065878ed226SJulian Elischer 		error = EINVAL;
1066878ed226SJulian Elischer 	} else /* Notify upper layer */
1067878ed226SJulian Elischer 		error = ng_hci_lp_qos_cfm(con, ep->status);
1068878ed226SJulian Elischer 
1069878ed226SJulian Elischer 	NG_FREE_M(event);
1070878ed226SJulian Elischer 
1071878ed226SJulian Elischer 	return (error);
1072878ed226SJulian Elischer } /* qos_setup_compl */
1073878ed226SJulian Elischer 
1074878ed226SJulian Elischer /* Hardware error event */
1075878ed226SJulian Elischer static int
1076878ed226SJulian Elischer hardware_error(ng_hci_unit_p unit, struct mbuf *event)
1077878ed226SJulian Elischer {
1078878ed226SJulian Elischer 	NG_HCI_ALERT(
1079878ed226SJulian Elischer "%s: %s - hardware error %#x\n",
1080878ed226SJulian Elischer 		__func__, NG_NODE_NAME(unit->node), *mtod(event, u_int8_t *));
1081878ed226SJulian Elischer 
1082878ed226SJulian Elischer 	NG_FREE_M(event);
1083878ed226SJulian Elischer 
1084878ed226SJulian Elischer 	return (0);
1085878ed226SJulian Elischer } /* hardware_error */
1086878ed226SJulian Elischer 
1087878ed226SJulian Elischer /* Role change event */
1088878ed226SJulian Elischer static int
1089878ed226SJulian Elischer role_change(ng_hci_unit_p unit, struct mbuf *event)
1090878ed226SJulian Elischer {
1091878ed226SJulian Elischer 	ng_hci_role_change_ep	*ep = NULL;
1092878ed226SJulian Elischer 	ng_hci_unit_con_p	 con = NULL;
1093878ed226SJulian Elischer 
1094878ed226SJulian Elischer 	NG_HCI_M_PULLUP(event, sizeof(*ep));
1095878ed226SJulian Elischer 	if (event == NULL)
1096878ed226SJulian Elischer 		return (ENOBUFS);
1097878ed226SJulian Elischer 
1098878ed226SJulian Elischer 	ep = mtod(event, ng_hci_role_change_ep *);
1099878ed226SJulian Elischer 
1100878ed226SJulian Elischer 	if (ep->status == 0) {
1101878ed226SJulian Elischer 		/* XXX shoud we also change "role" for SCO connections? */
1102878ed226SJulian Elischer 		con = ng_hci_con_by_bdaddr(unit, &ep->bdaddr, NG_HCI_LINK_ACL);
1103878ed226SJulian Elischer 		if (con != NULL)
1104878ed226SJulian Elischer 			con->role = ep->role;
1105878ed226SJulian Elischer 		else
1106878ed226SJulian Elischer 			NG_HCI_ALERT(
1107878ed226SJulian Elischer "%s: %s - ACL connection does not exist, bdaddr=%x:%x:%x:%x:%x:%x\n",
1108878ed226SJulian Elischer 				__func__, NG_NODE_NAME(unit->node),
1109878ed226SJulian Elischer 				ep->bdaddr.b[5], ep->bdaddr.b[4],
1110878ed226SJulian Elischer 				ep->bdaddr.b[3], ep->bdaddr.b[2],
1111878ed226SJulian Elischer 				ep->bdaddr.b[1], ep->bdaddr.b[0]);
1112878ed226SJulian Elischer 	} else
1113878ed226SJulian Elischer 		NG_HCI_ERR(
1114878ed226SJulian Elischer "%s: %s - failed to change role, status=%d, bdaddr=%x:%x:%x:%x:%x:%x\n",
1115878ed226SJulian Elischer 			__func__, NG_NODE_NAME(unit->node), ep->status,
1116878ed226SJulian Elischer 			ep->bdaddr.b[5], ep->bdaddr.b[4], ep->bdaddr.b[3],
1117878ed226SJulian Elischer 			ep->bdaddr.b[2], ep->bdaddr.b[1], ep->bdaddr.b[0]);
1118878ed226SJulian Elischer 
1119878ed226SJulian Elischer 	NG_FREE_M(event);
1120878ed226SJulian Elischer 
1121878ed226SJulian Elischer 	return (0);
1122878ed226SJulian Elischer } /* role_change */
1123878ed226SJulian Elischer 
1124878ed226SJulian Elischer /* Number of completed packets event */
1125878ed226SJulian Elischer static int
1126878ed226SJulian Elischer num_compl_pkts(ng_hci_unit_p unit, struct mbuf *event)
1127878ed226SJulian Elischer {
1128878ed226SJulian Elischer 	ng_hci_num_compl_pkts_ep	*ep = NULL;
1129878ed226SJulian Elischer 	ng_hci_unit_con_p		 con = NULL;
1130878ed226SJulian Elischer 	u_int16_t			 h, p;
1131878ed226SJulian Elischer 
1132878ed226SJulian Elischer 	NG_HCI_M_PULLUP(event, sizeof(*ep));
1133878ed226SJulian Elischer 	if (event == NULL)
1134878ed226SJulian Elischer 		return (ENOBUFS);
1135878ed226SJulian Elischer 
1136878ed226SJulian Elischer 	ep = mtod(event, ng_hci_num_compl_pkts_ep *);
1137878ed226SJulian Elischer 	m_adj(event, sizeof(*ep));
1138878ed226SJulian Elischer 
1139878ed226SJulian Elischer 	for (; ep->num_con_handles > 0; ep->num_con_handles --) {
1140878ed226SJulian Elischer 		/* Get connection handle */
1141878ed226SJulian Elischer 		m_copydata(event, 0, sizeof(h), (caddr_t) &h);
1142878ed226SJulian Elischer 		m_adj(event, sizeof(h));
1143878ed226SJulian Elischer 		h = NG_HCI_CON_HANDLE(le16toh(h));
1144878ed226SJulian Elischer 
1145878ed226SJulian Elischer 		/* Get number of completed packets */
1146878ed226SJulian Elischer 		m_copydata(event, 0, sizeof(p), (caddr_t) &p);
1147878ed226SJulian Elischer 		m_adj(event, sizeof(p));
1148878ed226SJulian Elischer 		p = le16toh(p);
1149878ed226SJulian Elischer 
1150878ed226SJulian Elischer 		/* Check if we have this connection handle */
1151878ed226SJulian Elischer 		con = ng_hci_con_by_handle(unit, h);
1152878ed226SJulian Elischer 		if (con != NULL) {
1153878ed226SJulian Elischer 			con->pending -= p;
1154878ed226SJulian Elischer 			if (con->pending < 0) {
1155878ed226SJulian Elischer 				NG_HCI_WARN(
1156878ed226SJulian Elischer "%s: %s - pending packet counter is out of sync! " \
1157878ed226SJulian Elischer "handle=%d, pending=%d, ncp=%d\n",	__func__, NG_NODE_NAME(unit->node),
1158878ed226SJulian Elischer 					con->con_handle, con->pending, p);
1159878ed226SJulian Elischer 
1160878ed226SJulian Elischer 				con->pending = 0;
1161878ed226SJulian Elischer 			}
1162878ed226SJulian Elischer 
1163878ed226SJulian Elischer 			/* Update buffer descriptor */
1164fbc48c2bSTakanori Watanabe 			if (con->link_type != NG_HCI_LINK_SCO)
1165878ed226SJulian Elischer 				NG_HCI_BUFF_ACL_FREE(unit->buffer, p);
1166878ed226SJulian Elischer 			else
1167878ed226SJulian Elischer 				NG_HCI_BUFF_SCO_FREE(unit->buffer, p);
1168878ed226SJulian Elischer 		} else
1169878ed226SJulian Elischer 			NG_HCI_ALERT(
1170878ed226SJulian Elischer "%s: %s - invalid connection handle=%d\n",
1171878ed226SJulian Elischer 				__func__, NG_NODE_NAME(unit->node), h);
1172878ed226SJulian Elischer 	}
1173878ed226SJulian Elischer 
1174878ed226SJulian Elischer 	NG_FREE_M(event);
1175878ed226SJulian Elischer 
1176878ed226SJulian Elischer 	/* Send more data */
1177878ed226SJulian Elischer 	ng_hci_send_data(unit);
1178878ed226SJulian Elischer 
1179878ed226SJulian Elischer 	return (0);
1180878ed226SJulian Elischer } /* num_compl_pkts */
1181878ed226SJulian Elischer 
1182878ed226SJulian Elischer /* Mode change event */
1183878ed226SJulian Elischer static int
1184878ed226SJulian Elischer mode_change(ng_hci_unit_p unit, struct mbuf *event)
1185878ed226SJulian Elischer {
1186878ed226SJulian Elischer 	ng_hci_mode_change_ep	*ep = NULL;
1187878ed226SJulian Elischer 	ng_hci_unit_con_p	 con = NULL;
1188878ed226SJulian Elischer 	int			 error = 0;
1189878ed226SJulian Elischer 
1190878ed226SJulian Elischer 	NG_HCI_M_PULLUP(event, sizeof(*ep));
1191878ed226SJulian Elischer 	if (event == NULL)
1192878ed226SJulian Elischer 		return (ENOBUFS);
1193878ed226SJulian Elischer 
1194878ed226SJulian Elischer 	ep = mtod(event, ng_hci_mode_change_ep *);
1195878ed226SJulian Elischer 
1196878ed226SJulian Elischer 	if (ep->status == 0) {
1197878ed226SJulian Elischer 		u_int16_t	h = NG_HCI_CON_HANDLE(le16toh(ep->con_handle));
1198878ed226SJulian Elischer 
1199878ed226SJulian Elischer 		con = ng_hci_con_by_handle(unit, h);
1200878ed226SJulian Elischer 		if (con == NULL) {
1201878ed226SJulian Elischer 			NG_HCI_ALERT(
1202878ed226SJulian Elischer "%s: %s - invalid connection handle=%d\n",
1203878ed226SJulian Elischer 				__func__, NG_NODE_NAME(unit->node), h);
1204878ed226SJulian Elischer 			error = ENOENT;
1205878ed226SJulian Elischer 		} else if (con->link_type != NG_HCI_LINK_ACL) {
1206878ed226SJulian Elischer 			NG_HCI_ALERT(
1207878ed226SJulian Elischer "%s: %s - invalid link type=%d\n",
1208878ed226SJulian Elischer 				__func__, NG_NODE_NAME(unit->node),
1209878ed226SJulian Elischer 				con->link_type);
1210878ed226SJulian Elischer 			error = EINVAL;
1211878ed226SJulian Elischer 		} else
1212878ed226SJulian Elischer 			con->mode = ep->unit_mode;
1213878ed226SJulian Elischer 	} else
1214878ed226SJulian Elischer 		NG_HCI_ERR(
1215878ed226SJulian Elischer "%s: %s - failed to change mode, status=%d\n",
1216878ed226SJulian Elischer 			__func__, NG_NODE_NAME(unit->node), ep->status);
1217878ed226SJulian Elischer 
1218878ed226SJulian Elischer 	NG_FREE_M(event);
1219878ed226SJulian Elischer 
1220878ed226SJulian Elischer 	return (error);
1221878ed226SJulian Elischer } /* mode_change */
1222878ed226SJulian Elischer 
1223878ed226SJulian Elischer /* Data buffer overflow event */
1224878ed226SJulian Elischer static int
1225878ed226SJulian Elischer data_buffer_overflow(ng_hci_unit_p unit, struct mbuf *event)
1226878ed226SJulian Elischer {
1227878ed226SJulian Elischer 	NG_HCI_ALERT(
1228878ed226SJulian Elischer "%s: %s - %s data buffer overflow\n",
1229878ed226SJulian Elischer 		__func__, NG_NODE_NAME(unit->node),
1230878ed226SJulian Elischer 		(*mtod(event, u_int8_t *) == NG_HCI_LINK_ACL)? "ACL" : "SCO");
1231878ed226SJulian Elischer 
1232878ed226SJulian Elischer 	NG_FREE_M(event);
1233878ed226SJulian Elischer 
1234878ed226SJulian Elischer 	return (0);
1235878ed226SJulian Elischer } /* data_buffer_overflow */
1236878ed226SJulian Elischer 
1237878ed226SJulian Elischer /* Read clock offset complete event */
1238878ed226SJulian Elischer static int
1239878ed226SJulian Elischer read_clock_offset_compl(ng_hci_unit_p unit, struct mbuf *event)
1240878ed226SJulian Elischer {
1241878ed226SJulian Elischer 	ng_hci_read_clock_offset_compl_ep	*ep = NULL;
1242878ed226SJulian Elischer 	ng_hci_unit_con_p			 con = NULL;
1243878ed226SJulian Elischer 	ng_hci_neighbor_p			 n = NULL;
1244878ed226SJulian Elischer 	int					 error = 0;
1245878ed226SJulian Elischer 
1246878ed226SJulian Elischer 	NG_HCI_M_PULLUP(event, sizeof(*ep));
1247878ed226SJulian Elischer 	if (event == NULL)
1248878ed226SJulian Elischer 		return (ENOBUFS);
1249878ed226SJulian Elischer 
1250878ed226SJulian Elischer 	ep = mtod(event, ng_hci_read_clock_offset_compl_ep *);
1251878ed226SJulian Elischer 
1252878ed226SJulian Elischer 	if (ep->status == 0) {
1253878ed226SJulian Elischer 		u_int16_t	h = NG_HCI_CON_HANDLE(le16toh(ep->con_handle));
1254878ed226SJulian Elischer 
1255878ed226SJulian Elischer 		con = ng_hci_con_by_handle(unit, h);
1256878ed226SJulian Elischer 		if (con == NULL) {
1257878ed226SJulian Elischer 			NG_HCI_ALERT(
1258878ed226SJulian Elischer "%s: %s - invalid connection handle=%d\n",
1259878ed226SJulian Elischer 				__func__, NG_NODE_NAME(unit->node), h);
1260878ed226SJulian Elischer 			error = ENOENT;
1261878ed226SJulian Elischer 			goto out;
1262878ed226SJulian Elischer 		}
1263878ed226SJulian Elischer 
1264878ed226SJulian Elischer 		/* Update cache entry */
1265fbc48c2bSTakanori Watanabe 		n = ng_hci_get_neighbor(unit, &con->bdaddr, NG_HCI_LINK_ACL);
1266878ed226SJulian Elischer 		if (n == NULL) {
1267878ed226SJulian Elischer 			n = ng_hci_new_neighbor(unit);
1268878ed226SJulian Elischer 			if (n == NULL) {
1269878ed226SJulian Elischer 				error = ENOMEM;
1270878ed226SJulian Elischer 				goto out;
1271878ed226SJulian Elischer 			}
1272878ed226SJulian Elischer 
1273878ed226SJulian Elischer 			bcopy(&con->bdaddr, &n->bdaddr, sizeof(n->bdaddr));
1274fbc48c2bSTakanori Watanabe 			n->addrtype = NG_HCI_LINK_ACL;
1275878ed226SJulian Elischer 		} else
1276878ed226SJulian Elischer 			getmicrotime(&n->updated);
1277878ed226SJulian Elischer 
1278878ed226SJulian Elischer 		n->clock_offset = le16toh(ep->clock_offset);
1279878ed226SJulian Elischer 	} else
1280878ed226SJulian Elischer 		NG_HCI_ERR(
1281878ed226SJulian Elischer "%s: %s - failed to Read Remote Clock Offset, status=%d\n",
1282878ed226SJulian Elischer 			__func__, NG_NODE_NAME(unit->node), ep->status);
1283878ed226SJulian Elischer out:
1284878ed226SJulian Elischer 	NG_FREE_M(event);
1285878ed226SJulian Elischer 
1286878ed226SJulian Elischer 	return (error);
1287878ed226SJulian Elischer } /* read_clock_offset_compl */
1288878ed226SJulian Elischer 
1289878ed226SJulian Elischer /* QoS violation event */
1290878ed226SJulian Elischer static int
1291878ed226SJulian Elischer qos_violation(ng_hci_unit_p unit, struct mbuf *event)
1292878ed226SJulian Elischer {
1293878ed226SJulian Elischer 	ng_hci_qos_violation_ep	*ep = NULL;
1294878ed226SJulian Elischer 	ng_hci_unit_con_p	 con = NULL;
1295878ed226SJulian Elischer 	u_int16_t		 h;
1296878ed226SJulian Elischer 	int			 error = 0;
1297878ed226SJulian Elischer 
1298878ed226SJulian Elischer 	NG_HCI_M_PULLUP(event, sizeof(*ep));
1299878ed226SJulian Elischer 	if (event == NULL)
1300878ed226SJulian Elischer 		return (ENOBUFS);
1301878ed226SJulian Elischer 
1302878ed226SJulian Elischer 	ep = mtod(event, ng_hci_qos_violation_ep *);
1303878ed226SJulian Elischer 
1304878ed226SJulian Elischer 	/* Check if we have this connection handle */
1305878ed226SJulian Elischer 	h = NG_HCI_CON_HANDLE(le16toh(ep->con_handle));
1306878ed226SJulian Elischer 	con = ng_hci_con_by_handle(unit, h);
1307878ed226SJulian Elischer 	if (con == NULL) {
1308878ed226SJulian Elischer 		NG_HCI_ALERT(
1309878ed226SJulian Elischer "%s: %s - invalid connection handle=%d\n",
1310878ed226SJulian Elischer 			__func__, NG_NODE_NAME(unit->node), h);
1311878ed226SJulian Elischer 		error = ENOENT;
1312878ed226SJulian Elischer 	} else if (con->link_type != NG_HCI_LINK_ACL) {
1313878ed226SJulian Elischer 		NG_HCI_ALERT(
1314878ed226SJulian Elischer "%s: %s - invalid link type=%d\n",
1315878ed226SJulian Elischer 			__func__, NG_NODE_NAME(unit->node), con->link_type);
1316878ed226SJulian Elischer 		error = EINVAL;
1317878ed226SJulian Elischer 	} else if (con->state != NG_HCI_CON_OPEN) {
1318878ed226SJulian Elischer 		NG_HCI_ALERT(
1319878ed226SJulian Elischer "%s: %s - invalid connection state=%d, handle=%d\n",
1320878ed226SJulian Elischer 			__func__, NG_NODE_NAME(unit->node), con->state, h);
1321878ed226SJulian Elischer 		error = EINVAL;
1322878ed226SJulian Elischer 	} else /* Notify upper layer */
1323878ed226SJulian Elischer 		error = ng_hci_lp_qos_ind(con);
1324878ed226SJulian Elischer 
1325878ed226SJulian Elischer 	NG_FREE_M(event);
1326878ed226SJulian Elischer 
1327878ed226SJulian Elischer 	return (error);
1328878ed226SJulian Elischer } /* qos_violation */
1329878ed226SJulian Elischer 
1330878ed226SJulian Elischer /* Page scan mode change event */
1331878ed226SJulian Elischer static int
1332878ed226SJulian Elischer page_scan_mode_change(ng_hci_unit_p unit, struct mbuf *event)
1333878ed226SJulian Elischer {
1334878ed226SJulian Elischer 	ng_hci_page_scan_mode_change_ep	*ep = NULL;
1335878ed226SJulian Elischer 	ng_hci_neighbor_p		 n = NULL;
1336878ed226SJulian Elischer 	int				 error = 0;
1337878ed226SJulian Elischer 
1338878ed226SJulian Elischer 	NG_HCI_M_PULLUP(event, sizeof(*ep));
1339878ed226SJulian Elischer 	if (event == NULL)
1340878ed226SJulian Elischer 		return (ENOBUFS);
1341878ed226SJulian Elischer 
1342878ed226SJulian Elischer 	ep = mtod(event, ng_hci_page_scan_mode_change_ep *);
1343878ed226SJulian Elischer 
1344878ed226SJulian Elischer 	/* Update cache entry */
1345fbc48c2bSTakanori Watanabe 	n = ng_hci_get_neighbor(unit, &ep->bdaddr, NG_HCI_LINK_ACL);
1346878ed226SJulian Elischer 	if (n == NULL) {
1347878ed226SJulian Elischer 		n = ng_hci_new_neighbor(unit);
1348878ed226SJulian Elischer 		if (n == NULL) {
1349878ed226SJulian Elischer 			error = ENOMEM;
1350878ed226SJulian Elischer 			goto out;
1351878ed226SJulian Elischer 		}
1352878ed226SJulian Elischer 
1353878ed226SJulian Elischer 		bcopy(&ep->bdaddr, &n->bdaddr, sizeof(n->bdaddr));
1354fbc48c2bSTakanori Watanabe 		n->addrtype = NG_HCI_LINK_ACL;
1355878ed226SJulian Elischer 	} else
1356878ed226SJulian Elischer 		getmicrotime(&n->updated);
1357878ed226SJulian Elischer 
1358878ed226SJulian Elischer 	n->page_scan_mode = ep->page_scan_mode;
1359878ed226SJulian Elischer out:
1360878ed226SJulian Elischer 	NG_FREE_M(event);
1361878ed226SJulian Elischer 
1362878ed226SJulian Elischer 	return (error);
1363878ed226SJulian Elischer } /* page_scan_mode_change */
1364878ed226SJulian Elischer 
1365878ed226SJulian Elischer /* Page scan repetition mode change event */
1366878ed226SJulian Elischer static int
1367878ed226SJulian Elischer page_scan_rep_mode_change(ng_hci_unit_p unit, struct mbuf *event)
1368878ed226SJulian Elischer {
1369878ed226SJulian Elischer 	ng_hci_page_scan_rep_mode_change_ep	*ep = NULL;
1370878ed226SJulian Elischer 	ng_hci_neighbor_p			 n = NULL;
1371878ed226SJulian Elischer 	int					 error = 0;
1372878ed226SJulian Elischer 
1373878ed226SJulian Elischer 	NG_HCI_M_PULLUP(event, sizeof(*ep));
1374878ed226SJulian Elischer 	if (event == NULL)
1375878ed226SJulian Elischer 		return (ENOBUFS);
1376878ed226SJulian Elischer 
1377878ed226SJulian Elischer 	ep = mtod(event, ng_hci_page_scan_rep_mode_change_ep *);
1378878ed226SJulian Elischer 
1379878ed226SJulian Elischer 	/* Update cache entry */
1380fbc48c2bSTakanori Watanabe 	n = ng_hci_get_neighbor(unit, &ep->bdaddr, NG_HCI_LINK_ACL);
1381878ed226SJulian Elischer 	if (n == NULL) {
1382878ed226SJulian Elischer 		n = ng_hci_new_neighbor(unit);
1383878ed226SJulian Elischer 		if (n == NULL) {
1384878ed226SJulian Elischer 			error = ENOMEM;
1385878ed226SJulian Elischer 			goto out;
1386878ed226SJulian Elischer 		}
1387878ed226SJulian Elischer 
1388878ed226SJulian Elischer 		bcopy(&ep->bdaddr, &n->bdaddr, sizeof(n->bdaddr));
1389fbc48c2bSTakanori Watanabe 		n->addrtype = NG_HCI_LINK_ACL;
1390878ed226SJulian Elischer 	} else
1391878ed226SJulian Elischer 		getmicrotime(&n->updated);
1392878ed226SJulian Elischer 
1393878ed226SJulian Elischer 	n->page_scan_rep_mode = ep->page_scan_rep_mode;
1394878ed226SJulian Elischer out:
1395878ed226SJulian Elischer 	NG_FREE_M(event);
1396878ed226SJulian Elischer 
1397878ed226SJulian Elischer 	return (error);
1398878ed226SJulian Elischer } /* page_scan_rep_mode_change */
1399