xref: /freebsd/sys/netgraph/bluetooth/hci/ng_hci_evnt.c (revision f2bb1cae36283a8eb5a0f19d8612c6abc5148e8f)
1878ed226SJulian Elischer /*
2878ed226SJulian Elischer  * ng_hci_evnt.c
3878ed226SJulian Elischer  *
4878ed226SJulian Elischer  * Copyright (c) Maksim Yevmenkin <m_evmenkin@yahoo.com>
5878ed226SJulian Elischer  * All rights reserved.
6878ed226SJulian Elischer  *
7878ed226SJulian Elischer  * Redistribution and use in source and binary forms, with or without
8878ed226SJulian Elischer  * modification, are permitted provided that the following conditions
9878ed226SJulian Elischer  * are met:
10878ed226SJulian Elischer  * 1. Redistributions of source code must retain the above copyright
11878ed226SJulian Elischer  *    notice, this list of conditions and the following disclaimer.
12878ed226SJulian Elischer  * 2. Redistributions in binary form must reproduce the above copyright
13878ed226SJulian Elischer  *    notice, this list of conditions and the following disclaimer in the
14878ed226SJulian Elischer  *    documentation and/or other materials provided with the distribution.
15878ed226SJulian Elischer  *
16878ed226SJulian Elischer  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
17878ed226SJulian Elischer  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
18878ed226SJulian Elischer  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
19878ed226SJulian Elischer  * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
20878ed226SJulian Elischer  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
21878ed226SJulian Elischer  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
22878ed226SJulian Elischer  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
23878ed226SJulian Elischer  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
24878ed226SJulian Elischer  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
25878ed226SJulian Elischer  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
26878ed226SJulian Elischer  * SUCH DAMAGE.
27878ed226SJulian Elischer  *
28f2bb1caeSJulian Elischer  * $Id: ng_hci_evnt.c,v 1.5 2003/04/01 18:15:25 max Exp $
29878ed226SJulian Elischer  * $FreeBSD$
30878ed226SJulian Elischer  */
31878ed226SJulian Elischer 
32878ed226SJulian Elischer #include <sys/param.h>
33878ed226SJulian Elischer #include <sys/systm.h>
34878ed226SJulian Elischer #include <sys/kernel.h>
35878ed226SJulian Elischer #include <sys/endian.h>
36878ed226SJulian Elischer #include <sys/malloc.h>
37878ed226SJulian Elischer #include <sys/mbuf.h>
38878ed226SJulian Elischer #include <sys/queue.h>
39878ed226SJulian Elischer #include <netgraph/ng_message.h>
40878ed226SJulian Elischer #include <netgraph/netgraph.h>
41878ed226SJulian Elischer #include "ng_bluetooth.h"
42878ed226SJulian Elischer #include "ng_hci.h"
43878ed226SJulian Elischer #include "ng_hci_var.h"
44878ed226SJulian Elischer #include "ng_hci_cmds.h"
45878ed226SJulian Elischer #include "ng_hci_evnt.h"
46878ed226SJulian Elischer #include "ng_hci_ulpi.h"
47878ed226SJulian Elischer #include "ng_hci_misc.h"
48878ed226SJulian Elischer 
49878ed226SJulian Elischer /******************************************************************************
50878ed226SJulian Elischer  ******************************************************************************
51878ed226SJulian Elischer  **                     HCI event processing module
52878ed226SJulian Elischer  ******************************************************************************
53878ed226SJulian Elischer  ******************************************************************************/
54878ed226SJulian Elischer 
55878ed226SJulian Elischer /*
56878ed226SJulian Elischer  * Event processing routines
57878ed226SJulian Elischer  */
58878ed226SJulian Elischer 
59878ed226SJulian Elischer static int inquiry_result             (ng_hci_unit_p, struct mbuf *);
60878ed226SJulian Elischer static int con_compl                  (ng_hci_unit_p, struct mbuf *);
61878ed226SJulian Elischer static int con_req                    (ng_hci_unit_p, struct mbuf *);
62878ed226SJulian Elischer static int discon_compl               (ng_hci_unit_p, struct mbuf *);
63f2bb1caeSJulian Elischer static int encryption_change          (ng_hci_unit_p, struct mbuf *);
64878ed226SJulian Elischer static int read_remote_features_compl (ng_hci_unit_p, struct mbuf *);
65878ed226SJulian Elischer static int qos_setup_compl            (ng_hci_unit_p, struct mbuf *);
66878ed226SJulian Elischer static int hardware_error             (ng_hci_unit_p, struct mbuf *);
67878ed226SJulian Elischer static int role_change                (ng_hci_unit_p, struct mbuf *);
68878ed226SJulian Elischer static int num_compl_pkts             (ng_hci_unit_p, struct mbuf *);
69878ed226SJulian Elischer static int mode_change                (ng_hci_unit_p, struct mbuf *);
70878ed226SJulian Elischer static int data_buffer_overflow       (ng_hci_unit_p, struct mbuf *);
71878ed226SJulian Elischer static int read_clock_offset_compl    (ng_hci_unit_p, struct mbuf *);
72878ed226SJulian Elischer static int qos_violation              (ng_hci_unit_p, struct mbuf *);
73878ed226SJulian Elischer static int page_scan_mode_change      (ng_hci_unit_p, struct mbuf *);
74878ed226SJulian Elischer static int page_scan_rep_mode_change  (ng_hci_unit_p, struct mbuf *);
75878ed226SJulian Elischer static int sync_con_queue             (ng_hci_unit_p, ng_hci_unit_con_p, int);
76878ed226SJulian Elischer static int send_data_packets          (ng_hci_unit_p, int, int);
77878ed226SJulian Elischer 
78878ed226SJulian Elischer /*
79878ed226SJulian Elischer  * Process HCI event packet
80878ed226SJulian Elischer  */
81878ed226SJulian Elischer 
82878ed226SJulian Elischer int
83878ed226SJulian Elischer ng_hci_process_event(ng_hci_unit_p unit, struct mbuf *event)
84878ed226SJulian Elischer {
85878ed226SJulian Elischer 	ng_hci_event_pkt_t	*hdr = NULL;
86878ed226SJulian Elischer 	int			 error = 0;
87878ed226SJulian Elischer 
88878ed226SJulian Elischer 	/* Get event packet header */
89878ed226SJulian Elischer 	NG_HCI_M_PULLUP(event, sizeof(*hdr));
90878ed226SJulian Elischer 	if (event == NULL)
91878ed226SJulian Elischer 		return (ENOBUFS);
92878ed226SJulian Elischer 
93878ed226SJulian Elischer 	hdr = mtod(event, ng_hci_event_pkt_t *);
94878ed226SJulian Elischer 
95878ed226SJulian Elischer 	NG_HCI_INFO(
96878ed226SJulian Elischer "%s: %s - got HCI event=%#x, length=%d\n",
97878ed226SJulian Elischer 		__func__, NG_NODE_NAME(unit->node), hdr->event, hdr->length);
98878ed226SJulian Elischer 
99878ed226SJulian Elischer 	/* Get rid of event header and process event */
100878ed226SJulian Elischer 	m_adj(event, sizeof(*hdr));
101878ed226SJulian Elischer 
102878ed226SJulian Elischer 	switch (hdr->event) {
103878ed226SJulian Elischer 	case NG_HCI_EVENT_INQUIRY_COMPL:
104878ed226SJulian Elischer 	case NG_HCI_EVENT_RETURN_LINK_KEYS:
105878ed226SJulian Elischer 	case NG_HCI_EVENT_PIN_CODE_REQ:
106878ed226SJulian Elischer 	case NG_HCI_EVENT_LINK_KEY_REQ:
107878ed226SJulian Elischer 	case NG_HCI_EVENT_LINK_KEY_NOTIFICATION:
108878ed226SJulian Elischer 	case NG_HCI_EVENT_LOOPBACK_COMMAND:
109878ed226SJulian Elischer 	case NG_HCI_EVENT_AUTH_COMPL:
110878ed226SJulian Elischer 	case NG_HCI_EVENT_CHANGE_CON_LINK_KEY_COMPL:
111878ed226SJulian Elischer 	case NG_HCI_EVENT_MASTER_LINK_KEY_COMPL:
112878ed226SJulian Elischer 	case NG_HCI_EVENT_FLUSH_OCCUR:	/* XXX Do we have to handle it? */
113878ed226SJulian Elischer 	case NG_HCI_EVENT_MAX_SLOT_CHANGE:
114878ed226SJulian Elischer 	case NG_HCI_EVENT_CON_PKT_TYPE_CHANGED:
115878ed226SJulian Elischer 	case NG_HCI_EVENT_BT_LOGO:
116878ed226SJulian Elischer 	case NG_HCI_EVENT_VENDOR:
117878ed226SJulian Elischer 	case NG_HCI_EVENT_REMOTE_NAME_REQ_COMPL:
118878ed226SJulian Elischer 	case NG_HCI_EVENT_READ_REMOTE_VER_INFO_COMPL:
119878ed226SJulian Elischer 		/* These do not need post processing */
120878ed226SJulian Elischer 		NG_FREE_M(event);
121878ed226SJulian Elischer 		break;
122878ed226SJulian Elischer 
123878ed226SJulian Elischer 	case NG_HCI_EVENT_INQUIRY_RESULT:
124878ed226SJulian Elischer 		error = inquiry_result(unit, event);
125878ed226SJulian Elischer 		break;
126878ed226SJulian Elischer 
127878ed226SJulian Elischer 	case NG_HCI_EVENT_CON_COMPL:
128878ed226SJulian Elischer 		error = con_compl(unit, event);
129878ed226SJulian Elischer 		break;
130878ed226SJulian Elischer 
131878ed226SJulian Elischer 	case NG_HCI_EVENT_CON_REQ:
132878ed226SJulian Elischer 		error = con_req(unit, event);
133878ed226SJulian Elischer 		break;
134878ed226SJulian Elischer 
135878ed226SJulian Elischer 	case NG_HCI_EVENT_DISCON_COMPL:
136878ed226SJulian Elischer 		error = discon_compl(unit, event);
137878ed226SJulian Elischer 		break;
138878ed226SJulian Elischer 
139f2bb1caeSJulian Elischer 	case NG_HCI_EVENT_ENCRYPTION_CHANGE:
140f2bb1caeSJulian Elischer 		error = encryption_change(unit, event);
141f2bb1caeSJulian Elischer 		break;
142f2bb1caeSJulian Elischer 
143878ed226SJulian Elischer 	case NG_HCI_EVENT_READ_REMOTE_FEATURES_COMPL:
144878ed226SJulian Elischer 		error = read_remote_features_compl(unit, event);
145878ed226SJulian Elischer 		break;
146878ed226SJulian Elischer 
147878ed226SJulian Elischer 	case NG_HCI_EVENT_QOS_SETUP_COMPL:
148878ed226SJulian Elischer 		error = qos_setup_compl(unit, event);
149878ed226SJulian Elischer 		break;
150878ed226SJulian Elischer 
151878ed226SJulian Elischer 	case NG_HCI_EVENT_COMMAND_COMPL:
152878ed226SJulian Elischer 		error = ng_hci_process_command_complete(unit, event);
153878ed226SJulian Elischer 		break;
154878ed226SJulian Elischer 
155878ed226SJulian Elischer 	case NG_HCI_EVENT_COMMAND_STATUS:
156878ed226SJulian Elischer 		error = ng_hci_process_command_status(unit, event);
157878ed226SJulian Elischer 		break;
158878ed226SJulian Elischer 
159878ed226SJulian Elischer 	case NG_HCI_EVENT_HARDWARE_ERROR:
160878ed226SJulian Elischer 		error = hardware_error(unit, event);
161878ed226SJulian Elischer 		break;
162878ed226SJulian Elischer 
163878ed226SJulian Elischer 	case NG_HCI_EVENT_ROLE_CHANGE:
164878ed226SJulian Elischer 		error = role_change(unit, event);
165878ed226SJulian Elischer 		break;
166878ed226SJulian Elischer 
167878ed226SJulian Elischer 	case NG_HCI_EVENT_NUM_COMPL_PKTS:
168878ed226SJulian Elischer 		error = num_compl_pkts(unit, event);
169878ed226SJulian Elischer 		break;
170878ed226SJulian Elischer 
171878ed226SJulian Elischer 	case NG_HCI_EVENT_MODE_CHANGE:
172878ed226SJulian Elischer 		error = mode_change(unit, event);
173878ed226SJulian Elischer 		break;
174878ed226SJulian Elischer 
175878ed226SJulian Elischer 	case NG_HCI_EVENT_DATA_BUFFER_OVERFLOW:
176878ed226SJulian Elischer 		error = data_buffer_overflow(unit, event);
177878ed226SJulian Elischer 		break;
178878ed226SJulian Elischer 
179878ed226SJulian Elischer 	case NG_HCI_EVENT_READ_CLOCK_OFFSET_COMPL:
180878ed226SJulian Elischer 		error = read_clock_offset_compl(unit, event);
181878ed226SJulian Elischer 		break;
182878ed226SJulian Elischer 
183878ed226SJulian Elischer 	case NG_HCI_EVENT_QOS_VIOLATION:
184878ed226SJulian Elischer 		error = qos_violation(unit, event);
185878ed226SJulian Elischer 		break;
186878ed226SJulian Elischer 
187878ed226SJulian Elischer 	case NG_HCI_EVENT_PAGE_SCAN_MODE_CHANGE:
188878ed226SJulian Elischer 		error = page_scan_mode_change(unit, event);
189878ed226SJulian Elischer 		break;
190878ed226SJulian Elischer 
191878ed226SJulian Elischer 	case NG_HCI_EVENT_PAGE_SCAN_REP_MODE_CHANGE:
192878ed226SJulian Elischer 		error = page_scan_rep_mode_change(unit, event);
193878ed226SJulian Elischer 		break;
194878ed226SJulian Elischer 
195878ed226SJulian Elischer 	default:
196878ed226SJulian Elischer 		NG_FREE_M(event);
197878ed226SJulian Elischer 		error = EINVAL;
198878ed226SJulian Elischer 		break;
199878ed226SJulian Elischer 	}
200878ed226SJulian Elischer 
201878ed226SJulian Elischer 	return (error);
202878ed226SJulian Elischer } /* ng_hci_process_event */
203878ed226SJulian Elischer 
204878ed226SJulian Elischer /*
205878ed226SJulian Elischer  * Send ACL and/or SCO data to the unit driver
206878ed226SJulian Elischer  */
207878ed226SJulian Elischer 
208878ed226SJulian Elischer void
209878ed226SJulian Elischer ng_hci_send_data(ng_hci_unit_p unit)
210878ed226SJulian Elischer {
211878ed226SJulian Elischer 	int	count;
212878ed226SJulian Elischer 
213878ed226SJulian Elischer 	/* Send ACL data */
214878ed226SJulian Elischer 	NG_HCI_BUFF_ACL_AVAIL(unit->buffer, count);
215878ed226SJulian Elischer 
216878ed226SJulian Elischer 	NG_HCI_INFO(
217878ed226SJulian Elischer "%s: %s - sending ACL data packets, count=%d\n",
218878ed226SJulian Elischer 		__func__, NG_NODE_NAME(unit->node), count);
219878ed226SJulian Elischer 
220878ed226SJulian Elischer 	if (count > 0) {
221878ed226SJulian Elischer 		count = send_data_packets(unit, NG_HCI_LINK_ACL, count);
222878ed226SJulian Elischer 		NG_HCI_STAT_ACL_SENT(unit->stat, count);
223878ed226SJulian Elischer 		NG_HCI_BUFF_ACL_USE(unit->buffer, count);
224878ed226SJulian Elischer 	}
225878ed226SJulian Elischer 
226878ed226SJulian Elischer 	/* Send SCO data */
227878ed226SJulian Elischer 	NG_HCI_BUFF_SCO_AVAIL(unit->buffer, count);
228878ed226SJulian Elischer 
229878ed226SJulian Elischer 	NG_HCI_INFO(
230878ed226SJulian Elischer "%s: %s - sending SCO data packets, count=%d\n",
231878ed226SJulian Elischer 		__func__, NG_NODE_NAME(unit->node), count);
232878ed226SJulian Elischer 
233878ed226SJulian Elischer 	if (count > 0) {
234878ed226SJulian Elischer 		count = send_data_packets(unit, NG_HCI_LINK_SCO, count);
235878ed226SJulian Elischer 		NG_HCI_STAT_SCO_SENT(unit->stat, count);
236878ed226SJulian Elischer 		NG_HCI_BUFF_SCO_USE(unit->buffer, count);
237878ed226SJulian Elischer 	}
238878ed226SJulian Elischer } /* ng_hci_send_data */
239878ed226SJulian Elischer 
240878ed226SJulian Elischer /*
241878ed226SJulian Elischer  * Send data packets to the lower layer.
242878ed226SJulian Elischer  */
243878ed226SJulian Elischer 
244878ed226SJulian Elischer static int
245878ed226SJulian Elischer send_data_packets(ng_hci_unit_p unit, int link_type, int limit)
246878ed226SJulian Elischer {
247878ed226SJulian Elischer 	ng_hci_unit_con_p	con = NULL, winner = NULL;
248878ed226SJulian Elischer 	item_p			item = NULL;
249878ed226SJulian Elischer 	int			min_pending, total_sent, sent, error, v;
250878ed226SJulian Elischer 
251878ed226SJulian Elischer 	for (total_sent = 0; limit > 0; ) {
252878ed226SJulian Elischer 		min_pending = 0x0fffffff;
253878ed226SJulian Elischer 		winner = NULL;
254878ed226SJulian Elischer 
255878ed226SJulian Elischer 		/*
256878ed226SJulian Elischer 		 * Find the connection that has has data to send
257878ed226SJulian Elischer 		 * and the smallest number of pending packets
258878ed226SJulian Elischer 		 */
259878ed226SJulian Elischer 
260878ed226SJulian Elischer 		LIST_FOREACH(con, &unit->con_list, next) {
261878ed226SJulian Elischer 			if (con->link_type != link_type)
262878ed226SJulian Elischer 				continue;
263878ed226SJulian Elischer 			if (NG_BT_ITEMQ_LEN(&con->conq) == 0)
264878ed226SJulian Elischer 				continue;
265878ed226SJulian Elischer 
266878ed226SJulian Elischer 			if (con->pending < min_pending) {
267878ed226SJulian Elischer 				winner = con;
268878ed226SJulian Elischer 				min_pending = con->pending;
269878ed226SJulian Elischer 			}
270878ed226SJulian Elischer 		}
271878ed226SJulian Elischer 
272878ed226SJulian Elischer 	        if (winner == NULL)
273878ed226SJulian Elischer 			break;
274878ed226SJulian Elischer 
275878ed226SJulian Elischer 		/*
276878ed226SJulian Elischer 		 * OK, we have a winner now send as much packets as we can
277878ed226SJulian Elischer 		 * Count the number of packets we have sent and then sync
278878ed226SJulian Elischer 		 * winner connection queue.
279878ed226SJulian Elischer 		 */
280878ed226SJulian Elischer 
281878ed226SJulian Elischer 		for (sent = 0; limit > 0; limit --, total_sent ++, sent ++) {
282878ed226SJulian Elischer 			NG_BT_ITEMQ_DEQUEUE(&winner->conq, item);
283878ed226SJulian Elischer 			if (item == NULL)
284878ed226SJulian Elischer 				break;
285878ed226SJulian Elischer 
286878ed226SJulian Elischer 			NG_HCI_INFO(
287878ed226SJulian Elischer "%s: %s - sending data packet, handle=%d, len=%d\n",
288878ed226SJulian Elischer 				__func__, NG_NODE_NAME(unit->node),
289878ed226SJulian Elischer 				winner->con_handle, NGI_M(item)->m_pkthdr.len);
290878ed226SJulian Elischer 
291878ed226SJulian Elischer 			/* Check if driver hook still there */
292878ed226SJulian Elischer 			v = (unit->drv != NULL && NG_HOOK_IS_VALID(unit->drv));
293878ed226SJulian Elischer 			if (!v || (unit->state & NG_HCI_UNIT_READY) !=
294878ed226SJulian Elischer 					NG_HCI_UNIT_READY) {
295878ed226SJulian Elischer 				NG_HCI_ERR(
296878ed226SJulian Elischer "%s: %s - could not send data. Hook \"%s\" is %svalid, state=%#x\n",
297878ed226SJulian Elischer 					__func__, NG_NODE_NAME(unit->node),
298878ed226SJulian Elischer 					NG_HCI_HOOK_DRV, ((v)? "" : "not "),
299878ed226SJulian Elischer 					unit->state);
300878ed226SJulian Elischer 
301878ed226SJulian Elischer 				NG_FREE_ITEM(item);
302878ed226SJulian Elischer 				error = ENOTCONN;
303878ed226SJulian Elischer 			} else {
304878ed226SJulian Elischer 				v = NGI_M(item)->m_pkthdr.len;
305878ed226SJulian Elischer 
306878ed226SJulian Elischer 				/* Give packet to raw hook */
307878ed226SJulian Elischer 				ng_hci_mtap(unit, NGI_M(item));
308878ed226SJulian Elischer 
309878ed226SJulian Elischer 				/* ... and forward item to the driver */
310878ed226SJulian Elischer 				NG_FWD_ITEM_HOOK(error, item, unit->drv);
311878ed226SJulian Elischer 			}
312878ed226SJulian Elischer 
313878ed226SJulian Elischer 			if (error != 0) {
314878ed226SJulian Elischer 				NG_HCI_ERR(
315878ed226SJulian Elischer "%s: %s - could not send data packet, handle=%d, error=%d\n",
316878ed226SJulian Elischer 					__func__, NG_NODE_NAME(unit->node),
317878ed226SJulian Elischer 					winner->con_handle, error);
318878ed226SJulian Elischer 				break;
319878ed226SJulian Elischer 			}
320878ed226SJulian Elischer 
321878ed226SJulian Elischer 			winner->pending ++;
322878ed226SJulian Elischer 			NG_HCI_STAT_BYTES_SENT(unit->stat, v);
323878ed226SJulian Elischer 		}
324878ed226SJulian Elischer 
325878ed226SJulian Elischer 		/*
326878ed226SJulian Elischer 		 * Sync connection queue for the winner
327878ed226SJulian Elischer 		 */
328878ed226SJulian Elischer 
329878ed226SJulian Elischer 		sync_con_queue(unit, winner, sent);
330878ed226SJulian Elischer 	}
331878ed226SJulian Elischer 
332878ed226SJulian Elischer 	return (total_sent);
333878ed226SJulian Elischer } /* send_data_packets */
334878ed226SJulian Elischer 
335878ed226SJulian Elischer /*
336878ed226SJulian Elischer  * Send flow control messages to the upper layer
337878ed226SJulian Elischer  */
338878ed226SJulian Elischer 
339878ed226SJulian Elischer static int
340878ed226SJulian Elischer sync_con_queue(ng_hci_unit_p unit, ng_hci_unit_con_p con, int completed)
341878ed226SJulian Elischer {
342878ed226SJulian Elischer 	hook_p				 hook = NULL;
343878ed226SJulian Elischer 	struct ng_mesg			*msg = NULL;
344878ed226SJulian Elischer 	ng_hci_sync_con_queue_ep	*state = NULL;
345878ed226SJulian Elischer 	int				 error;
346878ed226SJulian Elischer 
347878ed226SJulian Elischer 	hook = (con->link_type == NG_HCI_LINK_ACL)? unit->acl : unit->sco;
348878ed226SJulian Elischer 	if (hook == NULL || NG_HOOK_NOT_VALID(hook))
349878ed226SJulian Elischer 		return (ENOTCONN);
350878ed226SJulian Elischer 
351878ed226SJulian Elischer 	NG_MKMESSAGE(msg, NGM_HCI_COOKIE, NGM_HCI_SYNC_CON_QUEUE,
352878ed226SJulian Elischer 		sizeof(*state), M_NOWAIT);
353878ed226SJulian Elischer 	if (msg == NULL)
354878ed226SJulian Elischer 		return (ENOMEM);
355878ed226SJulian Elischer 
356878ed226SJulian Elischer 	state = (ng_hci_sync_con_queue_ep *)(msg->data);
357878ed226SJulian Elischer 	state->con_handle = con->con_handle;
358878ed226SJulian Elischer 	state->completed = completed;
359878ed226SJulian Elischer 
360878ed226SJulian Elischer 	NG_SEND_MSG_HOOK(error, unit->node, msg, hook, NULL);
361878ed226SJulian Elischer 
362878ed226SJulian Elischer 	return (error);
363878ed226SJulian Elischer } /* sync_con_queue */
364878ed226SJulian Elischer 
365878ed226SJulian Elischer /* Inquiry result event */
366878ed226SJulian Elischer static int
367878ed226SJulian Elischer inquiry_result(ng_hci_unit_p unit, struct mbuf *event)
368878ed226SJulian Elischer {
369878ed226SJulian Elischer 	ng_hci_inquiry_result_ep	*ep = NULL;
370878ed226SJulian Elischer 	ng_hci_neighbor_p		 n = NULL;
371878ed226SJulian Elischer 	bdaddr_t			 bdaddr;
372878ed226SJulian Elischer 	int				 error = 0;
373878ed226SJulian Elischer 
374878ed226SJulian Elischer 	NG_HCI_M_PULLUP(event, sizeof(*ep));
375878ed226SJulian Elischer 	if (event == NULL)
376878ed226SJulian Elischer 		return (ENOBUFS);
377878ed226SJulian Elischer 
378878ed226SJulian Elischer 	ep = mtod(event, ng_hci_inquiry_result_ep *);
379878ed226SJulian Elischer 	m_adj(event, sizeof(*ep));
380878ed226SJulian Elischer 
381878ed226SJulian Elischer 	for (; ep->num_responses > 0; ep->num_responses --) {
382878ed226SJulian Elischer 		/* Get remote unit address */
383878ed226SJulian Elischer 		m_copydata(event, 0, sizeof(bdaddr), (caddr_t) &bdaddr);
384878ed226SJulian Elischer 		m_adj(event, sizeof(bdaddr));
385878ed226SJulian Elischer 
386878ed226SJulian Elischer 		/* Lookup entry in the cache */
387878ed226SJulian Elischer 		n = ng_hci_get_neighbor(unit, &bdaddr);
388878ed226SJulian Elischer 		if (n == NULL) {
389878ed226SJulian Elischer 			/* Create new entry */
390878ed226SJulian Elischer 			n = ng_hci_new_neighbor(unit);
391878ed226SJulian Elischer 			if (n == NULL) {
392878ed226SJulian Elischer 				error = ENOMEM;
393878ed226SJulian Elischer 				break;
394878ed226SJulian Elischer 			}
395878ed226SJulian Elischer 		} else
396878ed226SJulian Elischer 			getmicrotime(&n->updated);
397878ed226SJulian Elischer 
398878ed226SJulian Elischer 		bcopy(&bdaddr, &n->bdaddr, sizeof(n->bdaddr));
399878ed226SJulian Elischer 
400878ed226SJulian Elischer 		/* XXX call m_pullup here? */
401878ed226SJulian Elischer 
402878ed226SJulian Elischer 		n->page_scan_rep_mode = *mtod(event, u_int8_t *);
403878ed226SJulian Elischer 		m_adj(event, sizeof(u_int8_t));
404878ed226SJulian Elischer 
405878ed226SJulian Elischer 		/* page_scan_period_mode */
406878ed226SJulian Elischer 		m_adj(event, sizeof(u_int8_t));
407878ed226SJulian Elischer 
408878ed226SJulian Elischer 		n->page_scan_mode = *mtod(event, u_int8_t *);
409878ed226SJulian Elischer 		m_adj(event, sizeof(u_int8_t));
410878ed226SJulian Elischer 
411878ed226SJulian Elischer 		/* class */
412878ed226SJulian Elischer 		m_adj(event, NG_HCI_CLASS_SIZE);
413878ed226SJulian Elischer 
414878ed226SJulian Elischer 		/* clock offset */
415878ed226SJulian Elischer 		m_copydata(event, 0, sizeof(n->clock_offset),
416878ed226SJulian Elischer 			(caddr_t) &n->clock_offset);
417878ed226SJulian Elischer 		n->clock_offset = le16toh(n->clock_offset);
418878ed226SJulian Elischer 	}
419878ed226SJulian Elischer 
420878ed226SJulian Elischer 	NG_FREE_M(event);
421878ed226SJulian Elischer 
422878ed226SJulian Elischer 	return (error);
423878ed226SJulian Elischer } /* inquiry_result */
424878ed226SJulian Elischer 
425878ed226SJulian Elischer /* Connection complete event */
426878ed226SJulian Elischer static int
427878ed226SJulian Elischer con_compl(ng_hci_unit_p unit, struct mbuf *event)
428878ed226SJulian Elischer {
429878ed226SJulian Elischer 	ng_hci_con_compl_ep	*ep = NULL;
430878ed226SJulian Elischer 	ng_hci_unit_con_p	 con = NULL;
431878ed226SJulian Elischer 	int			 error = 0;
432878ed226SJulian Elischer 
433878ed226SJulian Elischer 	NG_HCI_M_PULLUP(event, sizeof(*ep));
434878ed226SJulian Elischer 	if (event == NULL)
435878ed226SJulian Elischer 		return (ENOBUFS);
436878ed226SJulian Elischer 
437878ed226SJulian Elischer 	ep = mtod(event, ng_hci_con_compl_ep *);
438878ed226SJulian Elischer 
439878ed226SJulian Elischer 	/*
440878ed226SJulian Elischer 	 * Find the first connection descriptor that matches the following:
441878ed226SJulian Elischer 	 *
442878ed226SJulian Elischer 	 * 1) con->link_type == ep->link_type
443878ed226SJulian Elischer 	 * 2) con->state == NG_HCI_CON_W4_CONN_COMPLETE
444878ed226SJulian Elischer 	 * 3) con->bdaddr == ep->bdaddr
445878ed226SJulian Elischer 	 */
446878ed226SJulian Elischer 
447878ed226SJulian Elischer 	LIST_FOREACH(con, &unit->con_list, next)
448878ed226SJulian Elischer 		if (con->link_type == ep->link_type &&
449878ed226SJulian Elischer 		    con->state == NG_HCI_CON_W4_CONN_COMPLETE &&
450878ed226SJulian Elischer 		    bcmp(&con->bdaddr, &ep->bdaddr, sizeof(bdaddr_t)) == 0)
451878ed226SJulian Elischer 			break;
452878ed226SJulian Elischer 
453878ed226SJulian Elischer 	/*
454878ed226SJulian Elischer 	 * Two possible cases:
455878ed226SJulian Elischer 	 *
456878ed226SJulian Elischer 	 * 1) We have found connection descriptor. That means upper layer has
457878ed226SJulian Elischer 	 *    requested this connection via LP_CON_REQ message
458878ed226SJulian Elischer 	 *
459878ed226SJulian Elischer 	 * 2) We do not have connection descriptor. That means upper layer
460878ed226SJulian Elischer 	 *    nas not requested this connection or (less likely) we gave up
461878ed226SJulian Elischer 	 *    on this connection (timeout). The most likely scenario is that
462878ed226SJulian Elischer 	 *    we have received Create_Connection/Add_SCO_Connection command
463878ed226SJulian Elischer 	 *    from the RAW hook
464878ed226SJulian Elischer 	 */
465878ed226SJulian Elischer 
466878ed226SJulian Elischer 	if (con == NULL) {
467878ed226SJulian Elischer 		if (ep->status != 0)
468878ed226SJulian Elischer 			goto out;
469878ed226SJulian Elischer 
470878ed226SJulian Elischer 		con = ng_hci_new_con(unit, ep->link_type);
471878ed226SJulian Elischer 		if (con == NULL) {
472878ed226SJulian Elischer 			error = ENOMEM;
473878ed226SJulian Elischer 			goto out;
474878ed226SJulian Elischer 		}
475878ed226SJulian Elischer 
476878ed226SJulian Elischer 		bcopy(&ep->bdaddr, &con->bdaddr, sizeof(con->bdaddr));
477878ed226SJulian Elischer 	} else
478878ed226SJulian Elischer 		ng_hci_con_untimeout(con);
479878ed226SJulian Elischer 
480878ed226SJulian Elischer 	/*
481878ed226SJulian Elischer 	 * Update connection descriptor and send notification
482878ed226SJulian Elischer 	 * to the upper layers.
483878ed226SJulian Elischer 	 */
484878ed226SJulian Elischer 
485878ed226SJulian Elischer 	con->con_handle = NG_HCI_CON_HANDLE(le16toh(ep->con_handle));
486878ed226SJulian Elischer 	con->encryption_mode = ep->encryption_mode;
487878ed226SJulian Elischer 
488878ed226SJulian Elischer 	ng_hci_lp_con_cfm(con, ep->status);
489878ed226SJulian Elischer 
490878ed226SJulian Elischer 	/* Adjust connection state */
491878ed226SJulian Elischer 	if (ep->status != 0)
492878ed226SJulian Elischer 		ng_hci_free_con(con);
493878ed226SJulian Elischer 	else {
494878ed226SJulian Elischer 		con->state = NG_HCI_CON_OPEN;
495878ed226SJulian Elischer 
496878ed226SJulian Elischer 		/*
497878ed226SJulian Elischer 		 * Change link policy for the ACL connections. Enable all
498878ed226SJulian Elischer 		 * supported link modes. Enable Role switch as well if
499878ed226SJulian Elischer 		 * device supports it.
500878ed226SJulian Elischer 		 */
501878ed226SJulian Elischer 
502878ed226SJulian Elischer 		if (ep->link_type == NG_HCI_LINK_ACL) {
503878ed226SJulian Elischer 			struct __link_policy {
504878ed226SJulian Elischer 				ng_hci_cmd_pkt_t			 hdr;
505878ed226SJulian Elischer 				ng_hci_write_link_policy_settings_cp	 cp;
506878ed226SJulian Elischer 			} __attribute__ ((packed))			*lp;
507878ed226SJulian Elischer 			struct mbuf					*m;
508878ed226SJulian Elischer 
509a163d034SWarner Losh 			MGETHDR(m, M_DONTWAIT, MT_DATA);
510878ed226SJulian Elischer 			if (m != NULL) {
511878ed226SJulian Elischer 				m->m_pkthdr.len = m->m_len = sizeof(*lp);
512878ed226SJulian Elischer 				lp = mtod(m, struct __link_policy *);
513878ed226SJulian Elischer 
514878ed226SJulian Elischer 				lp->hdr.type = NG_HCI_CMD_PKT;
515878ed226SJulian Elischer 				lp->hdr.opcode = htole16(NG_HCI_OPCODE(
516878ed226SJulian Elischer 					NG_HCI_OGF_LINK_POLICY,
517878ed226SJulian Elischer 					NG_HCI_OCF_WRITE_LINK_POLICY_SETTINGS));
518878ed226SJulian Elischer 				lp->hdr.length = sizeof(lp->cp);
519878ed226SJulian Elischer 
520878ed226SJulian Elischer 				lp->cp.con_handle = ep->con_handle;
521878ed226SJulian Elischer 
522878ed226SJulian Elischer 				lp->cp.settings = 0;
523f2bb1caeSJulian Elischer 				if ((unit->features[0] & NG_HCI_LMP_SWITCH) &&
524f2bb1caeSJulian Elischer 				    unit->role_switch)
525878ed226SJulian Elischer 					lp->cp.settings |= 0x1;
526878ed226SJulian Elischer 				if (unit->features[0] & NG_HCI_LMP_HOLD_MODE)
527878ed226SJulian Elischer 					lp->cp.settings |= 0x2;
528878ed226SJulian Elischer 				if (unit->features[0] & NG_HCI_LMP_SNIFF_MODE)
529878ed226SJulian Elischer 					lp->cp.settings |= 0x4;
530878ed226SJulian Elischer 				if (unit->features[1] & NG_HCI_LMP_PARK_MODE)
531878ed226SJulian Elischer 					lp->cp.settings |= 0x8;
532878ed226SJulian Elischer 
533878ed226SJulian Elischer 				lp->cp.settings &= unit->link_policy_mask;
534878ed226SJulian Elischer 				lp->cp.settings = htole16(lp->cp.settings);
535878ed226SJulian Elischer 
536878ed226SJulian Elischer 				NG_BT_MBUFQ_ENQUEUE(&unit->cmdq, m);
537878ed226SJulian Elischer 				if (!(unit->state & NG_HCI_UNIT_COMMAND_PENDING))
538878ed226SJulian Elischer 					ng_hci_send_command(unit);
539878ed226SJulian Elischer 			}
540878ed226SJulian Elischer 		}
541878ed226SJulian Elischer 	}
542878ed226SJulian Elischer out:
543878ed226SJulian Elischer 	NG_FREE_M(event);
544878ed226SJulian Elischer 
545878ed226SJulian Elischer 	return (error);
546878ed226SJulian Elischer } /* com_compl */
547878ed226SJulian Elischer 
548878ed226SJulian Elischer /* Connection request event */
549878ed226SJulian Elischer static int
550878ed226SJulian Elischer con_req(ng_hci_unit_p unit, struct mbuf *event)
551878ed226SJulian Elischer {
552878ed226SJulian Elischer 	ng_hci_con_req_ep	*ep = NULL;
553878ed226SJulian Elischer 	ng_hci_unit_con_p	 con = NULL;
554878ed226SJulian Elischer 	int			 error = 0;
555878ed226SJulian Elischer 
556878ed226SJulian Elischer 	NG_HCI_M_PULLUP(event, sizeof(*ep));
557878ed226SJulian Elischer 	if (event == NULL)
558878ed226SJulian Elischer 		return (ENOBUFS);
559878ed226SJulian Elischer 
560878ed226SJulian Elischer 	ep = mtod(event, ng_hci_con_req_ep *);
561878ed226SJulian Elischer 
562878ed226SJulian Elischer 	/*
563878ed226SJulian Elischer 	 * Find the first connection descriptor that matches the following:
564878ed226SJulian Elischer 	 *
565878ed226SJulian Elischer 	 * 1) con->link_type == ep->link_type
566878ed226SJulian Elischer 	 *
567878ed226SJulian Elischer 	 * 2) con->state == NG_HCI_CON_W4_LP_CON_RSP ||
568878ed226SJulian Elischer 	 *    con->state == NG_HCI_CON_W4_CONN_COMPL
569878ed226SJulian Elischer 	 *
570878ed226SJulian Elischer 	 * 3) con->bdaddr == ep->bdaddr
571878ed226SJulian Elischer 	 *
572878ed226SJulian Elischer 	 * Possible cases:
573878ed226SJulian Elischer 	 *
574878ed226SJulian Elischer 	 * 1) We do not have connection descriptor. This is simple. Create
575878ed226SJulian Elischer 	 *    new fresh connection descriptor and send notification to the
576878ed226SJulian Elischer 	 *    appropriate upstream hook (based on link_type).
577878ed226SJulian Elischer 	 *
578878ed226SJulian Elischer 	 * 2) We found connection handle. This is more complicated.
579878ed226SJulian Elischer 	 *
580878ed226SJulian Elischer 	 * 2.1) ACL links
581878ed226SJulian Elischer 	 *
582878ed226SJulian Elischer 	 *      Since only one ACL link can exist between each pair of
583878ed226SJulian Elischer 	 *      units then we have a race. Our upper layer has requested
584878ed226SJulian Elischer 	 *      an ACL connection to the remote unit, but we did not send
585878ed226SJulian Elischer 	 *      command yet. At the same time the remote unit has requested
586878ed226SJulian Elischer 	 *      an ACL connection from us. In this case we will ignore
587878ed226SJulian Elischer 	 *	Connection_Request event. This probably will cause connect
588878ed226SJulian Elischer 	 *      failure	on both units.
589878ed226SJulian Elischer 	 *
590878ed226SJulian Elischer 	 * 2.2) SCO links
591878ed226SJulian Elischer 	 *
592878ed226SJulian Elischer 	 *      The spec on page 45 says :
593878ed226SJulian Elischer 	 *
594878ed226SJulian Elischer 	 *      "The master can support up to three SCO links to the same
595878ed226SJulian Elischer 	 *       slave or to different slaves. A slave can support up to
596878ed226SJulian Elischer 	 *       three SCO links from the same master, or two SCO links if
597878ed226SJulian Elischer 	 *       the links originate from different masters."
598878ed226SJulian Elischer 	 *
599878ed226SJulian Elischer 	 *      The only problem is how to handle multiple SCO links between
600878ed226SJulian Elischer 	 *      matster and slave. For now we will assume that multiple SCO
601878ed226SJulian Elischer 	 *      links MUST be opened one after another.
602878ed226SJulian Elischer 	 */
603878ed226SJulian Elischer 
604878ed226SJulian Elischer 	LIST_FOREACH(con, &unit->con_list, next)
605878ed226SJulian Elischer 		if (con->link_type == ep->link_type &&
606878ed226SJulian Elischer 		    (con->state == NG_HCI_CON_W4_LP_CON_RSP ||
607878ed226SJulian Elischer 		     con->state == NG_HCI_CON_W4_CONN_COMPLETE) &&
608878ed226SJulian Elischer 		    bcmp(&con->bdaddr, &ep->bdaddr, sizeof(bdaddr_t)) == 0)
609878ed226SJulian Elischer 			break;
610878ed226SJulian Elischer 
611878ed226SJulian Elischer 	if (con == NULL) {
612878ed226SJulian Elischer 		con = ng_hci_new_con(unit, ep->link_type);
613878ed226SJulian Elischer 		if (con != NULL) {
614878ed226SJulian Elischer 			bcopy(&ep->bdaddr, &con->bdaddr, sizeof(con->bdaddr));
615878ed226SJulian Elischer 
616878ed226SJulian Elischer 			con->state = NG_HCI_CON_W4_LP_CON_RSP;
617878ed226SJulian Elischer 			ng_hci_con_timeout(con);
618878ed226SJulian Elischer 
619878ed226SJulian Elischer 			error = ng_hci_lp_con_ind(con, ep->uclass);
620878ed226SJulian Elischer 			if (error != 0)
621878ed226SJulian Elischer 				ng_hci_free_con(con);
622878ed226SJulian Elischer 		} else
623878ed226SJulian Elischer 			error = ENOMEM;
624878ed226SJulian Elischer 	}
625878ed226SJulian Elischer 
626878ed226SJulian Elischer 	NG_FREE_M(event);
627878ed226SJulian Elischer 
628878ed226SJulian Elischer 	return (error);
629878ed226SJulian Elischer } /* con_req */
630878ed226SJulian Elischer 
631878ed226SJulian Elischer /* Disconnect complete event */
632878ed226SJulian Elischer static int
633878ed226SJulian Elischer discon_compl(ng_hci_unit_p unit, struct mbuf *event)
634878ed226SJulian Elischer {
635878ed226SJulian Elischer 	ng_hci_discon_compl_ep	*ep = NULL;
636878ed226SJulian Elischer 	ng_hci_unit_con_p	 con = NULL;
637878ed226SJulian Elischer 	int			 error = 0;
638878ed226SJulian Elischer 	u_int16_t		 h;
639878ed226SJulian Elischer 
640878ed226SJulian Elischer 	NG_HCI_M_PULLUP(event, sizeof(*ep));
641878ed226SJulian Elischer 	if (event == NULL)
642878ed226SJulian Elischer 		return (ENOBUFS);
643878ed226SJulian Elischer 
644878ed226SJulian Elischer 	ep = mtod(event, ng_hci_discon_compl_ep *);
645878ed226SJulian Elischer 
646878ed226SJulian Elischer 	/*
647878ed226SJulian Elischer 	 * XXX
648878ed226SJulian Elischer 	 * Do we have to send notification if ep->status != 0?
649878ed226SJulian Elischer 	 * For now we will send notification for both ACL and SCO connections
650878ed226SJulian Elischer 	 * ONLY if ep->status == 0.
651878ed226SJulian Elischer 	 */
652878ed226SJulian Elischer 
653878ed226SJulian Elischer 	if (ep->status == 0) {
654878ed226SJulian Elischer 		h = NG_HCI_CON_HANDLE(le16toh(ep->con_handle));
655878ed226SJulian Elischer 		con = ng_hci_con_by_handle(unit, h);
656878ed226SJulian Elischer 		if (con != NULL) {
657878ed226SJulian Elischer 			error = ng_hci_lp_discon_ind(con, ep->reason);
658878ed226SJulian Elischer 			ng_hci_free_con(con);
659878ed226SJulian Elischer 		} else {
660878ed226SJulian Elischer 			NG_HCI_ALERT(
661878ed226SJulian Elischer "%s: %s - invalid connection handle=%d\n",
662878ed226SJulian Elischer 				__func__, NG_NODE_NAME(unit->node), h);
663878ed226SJulian Elischer 			error = ENOENT;
664878ed226SJulian Elischer 		}
665878ed226SJulian Elischer 	}
666878ed226SJulian Elischer 
667878ed226SJulian Elischer 	NG_FREE_M(event);
668878ed226SJulian Elischer 
669878ed226SJulian Elischer 	return (error);
670878ed226SJulian Elischer } /* discon_compl */
671878ed226SJulian Elischer 
672f2bb1caeSJulian Elischer /* Encryption change event */
673f2bb1caeSJulian Elischer static int
674f2bb1caeSJulian Elischer encryption_change(ng_hci_unit_p unit, struct mbuf *event)
675f2bb1caeSJulian Elischer {
676f2bb1caeSJulian Elischer 	ng_hci_encryption_change_ep	*ep = NULL;
677f2bb1caeSJulian Elischer 	ng_hci_unit_con_p		 con = NULL;
678f2bb1caeSJulian Elischer 	int				 error = 0;
679f2bb1caeSJulian Elischer 
680f2bb1caeSJulian Elischer 	NG_HCI_M_PULLUP(event, sizeof(*ep));
681f2bb1caeSJulian Elischer 	if (event == NULL)
682f2bb1caeSJulian Elischer 		return (ENOBUFS);
683f2bb1caeSJulian Elischer 
684f2bb1caeSJulian Elischer 	ep = mtod(event, ng_hci_encryption_change_ep *);
685f2bb1caeSJulian Elischer 
686f2bb1caeSJulian Elischer 	if (ep->status == 0) {
687f2bb1caeSJulian Elischer 		u_int16_t	h = NG_HCI_CON_HANDLE(le16toh(ep->con_handle));
688f2bb1caeSJulian Elischer 
689f2bb1caeSJulian Elischer 		con = ng_hci_con_by_handle(unit, h);
690f2bb1caeSJulian Elischer 		if (con == NULL) {
691f2bb1caeSJulian Elischer 			NG_HCI_ALERT(
692f2bb1caeSJulian Elischer "%s: %s - invalid connection handle=%d\n",
693f2bb1caeSJulian Elischer 				__func__, NG_NODE_NAME(unit->node), h);
694f2bb1caeSJulian Elischer 			error = ENOENT;
695f2bb1caeSJulian Elischer 		} else if (con->link_type != NG_HCI_LINK_ACL) {
696f2bb1caeSJulian Elischer 			NG_HCI_ALERT(
697f2bb1caeSJulian Elischer "%s: %s - invalid link type=%d\n",
698f2bb1caeSJulian Elischer 				__func__, NG_NODE_NAME(unit->node),
699f2bb1caeSJulian Elischer 				con->link_type);
700f2bb1caeSJulian Elischer 			error = EINVAL;
701f2bb1caeSJulian Elischer 		} else if (ep->encryption_enable)
702f2bb1caeSJulian Elischer 			/* XXX is that true? */
703f2bb1caeSJulian Elischer 			con->encryption_mode = NG_HCI_ENCRYPTION_MODE_P2P;
704f2bb1caeSJulian Elischer 		else
705f2bb1caeSJulian Elischer 			con->encryption_mode = NG_HCI_ENCRYPTION_MODE_NONE;
706f2bb1caeSJulian Elischer 	} else
707f2bb1caeSJulian Elischer 		NG_HCI_ERR(
708f2bb1caeSJulian Elischer "%s: %s - failed to change encryption mode, status=%d\n",
709f2bb1caeSJulian Elischer 			__func__, NG_NODE_NAME(unit->node), ep->status);
710f2bb1caeSJulian Elischer 
711f2bb1caeSJulian Elischer 	NG_FREE_M(event);
712f2bb1caeSJulian Elischer 
713f2bb1caeSJulian Elischer 	return (error);
714f2bb1caeSJulian Elischer } /* encryption_change */
715f2bb1caeSJulian Elischer 
716878ed226SJulian Elischer /* Read remote feature complete event */
717878ed226SJulian Elischer static int
718878ed226SJulian Elischer read_remote_features_compl(ng_hci_unit_p unit, struct mbuf *event)
719878ed226SJulian Elischer {
720878ed226SJulian Elischer 	ng_hci_read_remote_features_compl_ep	*ep = NULL;
721878ed226SJulian Elischer 	ng_hci_unit_con_p			 con = NULL;
722878ed226SJulian Elischer 	ng_hci_neighbor_p			 n = NULL;
723878ed226SJulian Elischer 	u_int16_t				 h;
724878ed226SJulian Elischer 	int					 error = 0;
725878ed226SJulian Elischer 
726878ed226SJulian Elischer 	NG_HCI_M_PULLUP(event, sizeof(*ep));
727878ed226SJulian Elischer 	if (event == NULL)
728878ed226SJulian Elischer 		return (ENOBUFS);
729878ed226SJulian Elischer 
730878ed226SJulian Elischer 	ep = mtod(event, ng_hci_read_remote_features_compl_ep *);
731878ed226SJulian Elischer 
732878ed226SJulian Elischer 	if (ep->status == 0) {
733878ed226SJulian Elischer 		/* Check if we have this connection handle */
734878ed226SJulian Elischer 		h = NG_HCI_CON_HANDLE(le16toh(ep->con_handle));
735878ed226SJulian Elischer 		con = ng_hci_con_by_handle(unit, h);
736878ed226SJulian Elischer 		if (con == NULL) {
737878ed226SJulian Elischer 			NG_HCI_ALERT(
738878ed226SJulian Elischer "%s: %s - invalid connection handle=%d\n",
739878ed226SJulian Elischer 				__func__, NG_NODE_NAME(unit->node), h);
740878ed226SJulian Elischer 			error = ENOENT;
741878ed226SJulian Elischer 			goto out;
742878ed226SJulian Elischer 		}
743878ed226SJulian Elischer 
744878ed226SJulian Elischer 		/* Update cache entry */
745878ed226SJulian Elischer 		n = ng_hci_get_neighbor(unit, &con->bdaddr);
746878ed226SJulian Elischer 		if (n == NULL) {
747878ed226SJulian Elischer 			n = ng_hci_new_neighbor(unit);
748878ed226SJulian Elischer 			if (n == NULL) {
749878ed226SJulian Elischer 				error = ENOMEM;
750878ed226SJulian Elischer 				goto out;
751878ed226SJulian Elischer 			}
752878ed226SJulian Elischer 
753878ed226SJulian Elischer 			bcopy(&con->bdaddr, &n->bdaddr, sizeof(n->bdaddr));
754878ed226SJulian Elischer 		} else
755878ed226SJulian Elischer 			getmicrotime(&n->updated);
756878ed226SJulian Elischer 
757878ed226SJulian Elischer 		bcopy(ep->features, n->features, sizeof(n->features));
758878ed226SJulian Elischer 	} else
759878ed226SJulian Elischer 		NG_HCI_ERR(
760878ed226SJulian Elischer "%s: %s - failed to read remote unit features, status=%d\n",
761878ed226SJulian Elischer 			__func__, NG_NODE_NAME(unit->node), ep->status);
762878ed226SJulian Elischer out:
763878ed226SJulian Elischer 	NG_FREE_M(event);
764878ed226SJulian Elischer 
765878ed226SJulian Elischer 	return (error);
766878ed226SJulian Elischer } /* read_remote_features_compl */
767878ed226SJulian Elischer 
768878ed226SJulian Elischer /* QoS setup complete event */
769878ed226SJulian Elischer static int
770878ed226SJulian Elischer qos_setup_compl(ng_hci_unit_p unit, struct mbuf *event)
771878ed226SJulian Elischer {
772878ed226SJulian Elischer 	ng_hci_qos_setup_compl_ep	*ep = NULL;
773878ed226SJulian Elischer 	ng_hci_unit_con_p		 con = NULL;
774878ed226SJulian Elischer 	u_int16_t			 h;
775878ed226SJulian Elischer 	int				 error = 0;
776878ed226SJulian Elischer 
777878ed226SJulian Elischer 	NG_HCI_M_PULLUP(event, sizeof(*ep));
778878ed226SJulian Elischer 	if (event == NULL)
779878ed226SJulian Elischer 		return (ENOBUFS);
780878ed226SJulian Elischer 
781878ed226SJulian Elischer 	ep = mtod(event, ng_hci_qos_setup_compl_ep *);
782878ed226SJulian Elischer 
783878ed226SJulian Elischer 	/* Check if we have this connection handle */
784878ed226SJulian Elischer 	h = NG_HCI_CON_HANDLE(le16toh(ep->con_handle));
785878ed226SJulian Elischer 	con = ng_hci_con_by_handle(unit, h);
786878ed226SJulian Elischer 	if (con == NULL) {
787878ed226SJulian Elischer 		NG_HCI_ALERT(
788878ed226SJulian Elischer "%s: %s - invalid connection handle=%d\n",
789878ed226SJulian Elischer 			__func__, NG_NODE_NAME(unit->node), h);
790878ed226SJulian Elischer 		error = ENOENT;
791878ed226SJulian Elischer 	} else if (con->link_type != NG_HCI_LINK_ACL) {
792878ed226SJulian Elischer 		NG_HCI_ALERT(
793878ed226SJulian Elischer "%s: %s - invalid link type=%d, handle=%d\n",
794878ed226SJulian Elischer 			__func__, NG_NODE_NAME(unit->node), con->link_type, h);
795878ed226SJulian Elischer 		error = EINVAL;
796878ed226SJulian Elischer 	} else if (con->state != NG_HCI_CON_OPEN) {
797878ed226SJulian Elischer 		NG_HCI_ALERT(
798878ed226SJulian Elischer "%s: %s - invalid connection state=%d, handle=%d\n",
799878ed226SJulian Elischer 			__func__, NG_NODE_NAME(unit->node),
800878ed226SJulian Elischer 			con->state, h);
801878ed226SJulian Elischer 		error = EINVAL;
802878ed226SJulian Elischer 	} else /* Notify upper layer */
803878ed226SJulian Elischer 		error = ng_hci_lp_qos_cfm(con, ep->status);
804878ed226SJulian Elischer 
805878ed226SJulian Elischer 	NG_FREE_M(event);
806878ed226SJulian Elischer 
807878ed226SJulian Elischer 	return (error);
808878ed226SJulian Elischer } /* qos_setup_compl */
809878ed226SJulian Elischer 
810878ed226SJulian Elischer /* Hardware error event */
811878ed226SJulian Elischer static int
812878ed226SJulian Elischer hardware_error(ng_hci_unit_p unit, struct mbuf *event)
813878ed226SJulian Elischer {
814878ed226SJulian Elischer 	NG_HCI_ALERT(
815878ed226SJulian Elischer "%s: %s - hardware error %#x\n",
816878ed226SJulian Elischer 		__func__, NG_NODE_NAME(unit->node), *mtod(event, u_int8_t *));
817878ed226SJulian Elischer 
818878ed226SJulian Elischer 	NG_FREE_M(event);
819878ed226SJulian Elischer 
820878ed226SJulian Elischer 	return (0);
821878ed226SJulian Elischer } /* hardware_error */
822878ed226SJulian Elischer 
823878ed226SJulian Elischer /* Role change event */
824878ed226SJulian Elischer static int
825878ed226SJulian Elischer role_change(ng_hci_unit_p unit, struct mbuf *event)
826878ed226SJulian Elischer {
827878ed226SJulian Elischer 	ng_hci_role_change_ep	*ep = NULL;
828878ed226SJulian Elischer 	ng_hci_unit_con_p	 con = NULL;
829878ed226SJulian Elischer 
830878ed226SJulian Elischer 	NG_HCI_M_PULLUP(event, sizeof(*ep));
831878ed226SJulian Elischer 	if (event == NULL)
832878ed226SJulian Elischer 		return (ENOBUFS);
833878ed226SJulian Elischer 
834878ed226SJulian Elischer 	ep = mtod(event, ng_hci_role_change_ep *);
835878ed226SJulian Elischer 
836878ed226SJulian Elischer 	if (ep->status == 0) {
837878ed226SJulian Elischer 		/* XXX shoud we also change "role" for SCO connections? */
838878ed226SJulian Elischer 		con = ng_hci_con_by_bdaddr(unit, &ep->bdaddr, NG_HCI_LINK_ACL);
839878ed226SJulian Elischer 		if (con != NULL)
840878ed226SJulian Elischer 			con->role = ep->role;
841878ed226SJulian Elischer 		else
842878ed226SJulian Elischer 			NG_HCI_ALERT(
843878ed226SJulian Elischer "%s: %s - ACL connection does not exist, bdaddr=%x:%x:%x:%x:%x:%x\n",
844878ed226SJulian Elischer 				__func__, NG_NODE_NAME(unit->node),
845878ed226SJulian Elischer 				ep->bdaddr.b[5], ep->bdaddr.b[4],
846878ed226SJulian Elischer 				ep->bdaddr.b[3], ep->bdaddr.b[2],
847878ed226SJulian Elischer 				ep->bdaddr.b[1], ep->bdaddr.b[0]);
848878ed226SJulian Elischer 	} else
849878ed226SJulian Elischer 		NG_HCI_ERR(
850878ed226SJulian Elischer "%s: %s - failed to change role, status=%d, bdaddr=%x:%x:%x:%x:%x:%x\n",
851878ed226SJulian Elischer 			__func__, NG_NODE_NAME(unit->node), ep->status,
852878ed226SJulian Elischer 			ep->bdaddr.b[5], ep->bdaddr.b[4], ep->bdaddr.b[3],
853878ed226SJulian Elischer 			ep->bdaddr.b[2], ep->bdaddr.b[1], ep->bdaddr.b[0]);
854878ed226SJulian Elischer 
855878ed226SJulian Elischer 	NG_FREE_M(event);
856878ed226SJulian Elischer 
857878ed226SJulian Elischer 	return (0);
858878ed226SJulian Elischer } /* role_change */
859878ed226SJulian Elischer 
860878ed226SJulian Elischer /* Number of completed packets event */
861878ed226SJulian Elischer static int
862878ed226SJulian Elischer num_compl_pkts(ng_hci_unit_p unit, struct mbuf *event)
863878ed226SJulian Elischer {
864878ed226SJulian Elischer 	ng_hci_num_compl_pkts_ep	*ep = NULL;
865878ed226SJulian Elischer 	ng_hci_unit_con_p		 con = NULL;
866878ed226SJulian Elischer 	u_int16_t			 h, p;
867878ed226SJulian Elischer 
868878ed226SJulian Elischer 	NG_HCI_M_PULLUP(event, sizeof(*ep));
869878ed226SJulian Elischer 	if (event == NULL)
870878ed226SJulian Elischer 		return (ENOBUFS);
871878ed226SJulian Elischer 
872878ed226SJulian Elischer 	ep = mtod(event, ng_hci_num_compl_pkts_ep *);
873878ed226SJulian Elischer 	m_adj(event, sizeof(*ep));
874878ed226SJulian Elischer 
875878ed226SJulian Elischer 	for (; ep->num_con_handles > 0; ep->num_con_handles --) {
876878ed226SJulian Elischer 		/* Get connection handle */
877878ed226SJulian Elischer 		m_copydata(event, 0, sizeof(h), (caddr_t) &h);
878878ed226SJulian Elischer 		m_adj(event, sizeof(h));
879878ed226SJulian Elischer 		h = NG_HCI_CON_HANDLE(le16toh(h));
880878ed226SJulian Elischer 
881878ed226SJulian Elischer 		/* Get number of completed packets */
882878ed226SJulian Elischer 		m_copydata(event, 0, sizeof(p), (caddr_t) &p);
883878ed226SJulian Elischer 		m_adj(event, sizeof(p));
884878ed226SJulian Elischer 		p = le16toh(p);
885878ed226SJulian Elischer 
886878ed226SJulian Elischer 		/* Check if we have this connection handle */
887878ed226SJulian Elischer 		con = ng_hci_con_by_handle(unit, h);
888878ed226SJulian Elischer 		if (con != NULL) {
889878ed226SJulian Elischer 			con->pending -= p;
890878ed226SJulian Elischer 			if (con->pending < 0) {
891878ed226SJulian Elischer 				NG_HCI_WARN(
892878ed226SJulian Elischer "%s: %s - pending packet counter is out of sync! " \
893878ed226SJulian Elischer "handle=%d, pending=%d, ncp=%d\n",	__func__, NG_NODE_NAME(unit->node),
894878ed226SJulian Elischer 					con->con_handle, con->pending, p);
895878ed226SJulian Elischer 
896878ed226SJulian Elischer 				con->pending = 0;
897878ed226SJulian Elischer 			}
898878ed226SJulian Elischer 
899878ed226SJulian Elischer 			/* Update buffer descriptor */
900878ed226SJulian Elischer 			if (con->link_type == NG_HCI_LINK_ACL)
901878ed226SJulian Elischer 				NG_HCI_BUFF_ACL_FREE(unit->buffer, p);
902878ed226SJulian Elischer 			else
903878ed226SJulian Elischer 				NG_HCI_BUFF_SCO_FREE(unit->buffer, p);
904878ed226SJulian Elischer 		} else
905878ed226SJulian Elischer 			NG_HCI_ALERT(
906878ed226SJulian Elischer "%s: %s - invalid connection handle=%d\n",
907878ed226SJulian Elischer 				__func__, NG_NODE_NAME(unit->node), h);
908878ed226SJulian Elischer 	}
909878ed226SJulian Elischer 
910878ed226SJulian Elischer 	NG_FREE_M(event);
911878ed226SJulian Elischer 
912878ed226SJulian Elischer 	/* Send more data */
913878ed226SJulian Elischer 	ng_hci_send_data(unit);
914878ed226SJulian Elischer 
915878ed226SJulian Elischer 	return (0);
916878ed226SJulian Elischer } /* num_compl_pkts */
917878ed226SJulian Elischer 
918878ed226SJulian Elischer /* Mode change event */
919878ed226SJulian Elischer static int
920878ed226SJulian Elischer mode_change(ng_hci_unit_p unit, struct mbuf *event)
921878ed226SJulian Elischer {
922878ed226SJulian Elischer 	ng_hci_mode_change_ep	*ep = NULL;
923878ed226SJulian Elischer 	ng_hci_unit_con_p	 con = NULL;
924878ed226SJulian Elischer 	int			 error = 0;
925878ed226SJulian Elischer 
926878ed226SJulian Elischer 	NG_HCI_M_PULLUP(event, sizeof(*ep));
927878ed226SJulian Elischer 	if (event == NULL)
928878ed226SJulian Elischer 		return (ENOBUFS);
929878ed226SJulian Elischer 
930878ed226SJulian Elischer 	ep = mtod(event, ng_hci_mode_change_ep *);
931878ed226SJulian Elischer 
932878ed226SJulian Elischer 	if (ep->status == 0) {
933878ed226SJulian Elischer 		u_int16_t	h = NG_HCI_CON_HANDLE(le16toh(ep->con_handle));
934878ed226SJulian Elischer 
935878ed226SJulian Elischer 		con = ng_hci_con_by_handle(unit, h);
936878ed226SJulian Elischer 		if (con == NULL) {
937878ed226SJulian Elischer 			NG_HCI_ALERT(
938878ed226SJulian Elischer "%s: %s - invalid connection handle=%d\n",
939878ed226SJulian Elischer 				__func__, NG_NODE_NAME(unit->node), h);
940878ed226SJulian Elischer 			error = ENOENT;
941878ed226SJulian Elischer 		} else if (con->link_type != NG_HCI_LINK_ACL) {
942878ed226SJulian Elischer 			NG_HCI_ALERT(
943878ed226SJulian Elischer "%s: %s - invalid link type=%d\n",
944878ed226SJulian Elischer 				__func__, NG_NODE_NAME(unit->node),
945878ed226SJulian Elischer 				con->link_type);
946878ed226SJulian Elischer 			error = EINVAL;
947878ed226SJulian Elischer 		} else
948878ed226SJulian Elischer 			con->mode = ep->unit_mode;
949878ed226SJulian Elischer 	} else
950878ed226SJulian Elischer 		NG_HCI_ERR(
951878ed226SJulian Elischer "%s: %s - failed to change mode, status=%d\n",
952878ed226SJulian Elischer 			__func__, NG_NODE_NAME(unit->node), ep->status);
953878ed226SJulian Elischer 
954878ed226SJulian Elischer 	NG_FREE_M(event);
955878ed226SJulian Elischer 
956878ed226SJulian Elischer 	return (error);
957878ed226SJulian Elischer } /* mode_change */
958878ed226SJulian Elischer 
959878ed226SJulian Elischer /* Data buffer overflow event */
960878ed226SJulian Elischer static int
961878ed226SJulian Elischer data_buffer_overflow(ng_hci_unit_p unit, struct mbuf *event)
962878ed226SJulian Elischer {
963878ed226SJulian Elischer 	NG_HCI_ALERT(
964878ed226SJulian Elischer "%s: %s - %s data buffer overflow\n",
965878ed226SJulian Elischer 		__func__, NG_NODE_NAME(unit->node),
966878ed226SJulian Elischer 		(*mtod(event, u_int8_t *) == NG_HCI_LINK_ACL)? "ACL" : "SCO");
967878ed226SJulian Elischer 
968878ed226SJulian Elischer 	NG_FREE_M(event);
969878ed226SJulian Elischer 
970878ed226SJulian Elischer 	return (0);
971878ed226SJulian Elischer } /* data_buffer_overflow */
972878ed226SJulian Elischer 
973878ed226SJulian Elischer /* Read clock offset complete event */
974878ed226SJulian Elischer static int
975878ed226SJulian Elischer read_clock_offset_compl(ng_hci_unit_p unit, struct mbuf *event)
976878ed226SJulian Elischer {
977878ed226SJulian Elischer 	ng_hci_read_clock_offset_compl_ep	*ep = NULL;
978878ed226SJulian Elischer 	ng_hci_unit_con_p			 con = NULL;
979878ed226SJulian Elischer 	ng_hci_neighbor_p			 n = NULL;
980878ed226SJulian Elischer 	int					 error = 0;
981878ed226SJulian Elischer 
982878ed226SJulian Elischer 	NG_HCI_M_PULLUP(event, sizeof(*ep));
983878ed226SJulian Elischer 	if (event == NULL)
984878ed226SJulian Elischer 		return (ENOBUFS);
985878ed226SJulian Elischer 
986878ed226SJulian Elischer 	ep = mtod(event, ng_hci_read_clock_offset_compl_ep *);
987878ed226SJulian Elischer 
988878ed226SJulian Elischer 	if (ep->status == 0) {
989878ed226SJulian Elischer 		u_int16_t	h = NG_HCI_CON_HANDLE(le16toh(ep->con_handle));
990878ed226SJulian Elischer 
991878ed226SJulian Elischer 		con = ng_hci_con_by_handle(unit, h);
992878ed226SJulian Elischer 		if (con == NULL) {
993878ed226SJulian Elischer 			NG_HCI_ALERT(
994878ed226SJulian Elischer "%s: %s - invalid connection handle=%d\n",
995878ed226SJulian Elischer 				__func__, NG_NODE_NAME(unit->node), h);
996878ed226SJulian Elischer 			error = ENOENT;
997878ed226SJulian Elischer 			goto out;
998878ed226SJulian Elischer 		}
999878ed226SJulian Elischer 
1000878ed226SJulian Elischer 		/* Update cache entry */
1001878ed226SJulian Elischer 		n = ng_hci_get_neighbor(unit, &con->bdaddr);
1002878ed226SJulian Elischer 		if (n == NULL) {
1003878ed226SJulian Elischer 			n = ng_hci_new_neighbor(unit);
1004878ed226SJulian Elischer 			if (n == NULL) {
1005878ed226SJulian Elischer 				error = ENOMEM;
1006878ed226SJulian Elischer 				goto out;
1007878ed226SJulian Elischer 			}
1008878ed226SJulian Elischer 
1009878ed226SJulian Elischer 			bcopy(&con->bdaddr, &n->bdaddr, sizeof(n->bdaddr));
1010878ed226SJulian Elischer 		} else
1011878ed226SJulian Elischer 			getmicrotime(&n->updated);
1012878ed226SJulian Elischer 
1013878ed226SJulian Elischer 		n->clock_offset = le16toh(ep->clock_offset);
1014878ed226SJulian Elischer 	} else
1015878ed226SJulian Elischer 		NG_HCI_ERR(
1016878ed226SJulian Elischer "%s: %s - failed to Read Remote Clock Offset, status=%d\n",
1017878ed226SJulian Elischer 			__func__, NG_NODE_NAME(unit->node), ep->status);
1018878ed226SJulian Elischer out:
1019878ed226SJulian Elischer 	NG_FREE_M(event);
1020878ed226SJulian Elischer 
1021878ed226SJulian Elischer 	return (error);
1022878ed226SJulian Elischer } /* read_clock_offset_compl */
1023878ed226SJulian Elischer 
1024878ed226SJulian Elischer /* QoS violation event */
1025878ed226SJulian Elischer static int
1026878ed226SJulian Elischer qos_violation(ng_hci_unit_p unit, struct mbuf *event)
1027878ed226SJulian Elischer {
1028878ed226SJulian Elischer 	ng_hci_qos_violation_ep	*ep = NULL;
1029878ed226SJulian Elischer 	ng_hci_unit_con_p	 con = NULL;
1030878ed226SJulian Elischer 	u_int16_t		 h;
1031878ed226SJulian Elischer 	int			 error = 0;
1032878ed226SJulian Elischer 
1033878ed226SJulian Elischer 	NG_HCI_M_PULLUP(event, sizeof(*ep));
1034878ed226SJulian Elischer 	if (event == NULL)
1035878ed226SJulian Elischer 		return (ENOBUFS);
1036878ed226SJulian Elischer 
1037878ed226SJulian Elischer 	ep = mtod(event, ng_hci_qos_violation_ep *);
1038878ed226SJulian Elischer 
1039878ed226SJulian Elischer 	/* Check if we have this connection handle */
1040878ed226SJulian Elischer 	h = NG_HCI_CON_HANDLE(le16toh(ep->con_handle));
1041878ed226SJulian Elischer 	con = ng_hci_con_by_handle(unit, h);
1042878ed226SJulian Elischer 	if (con == NULL) {
1043878ed226SJulian Elischer 		NG_HCI_ALERT(
1044878ed226SJulian Elischer "%s: %s - invalid connection handle=%d\n",
1045878ed226SJulian Elischer 			__func__, NG_NODE_NAME(unit->node), h);
1046878ed226SJulian Elischer 		error = ENOENT;
1047878ed226SJulian Elischer 	} else if (con->link_type != NG_HCI_LINK_ACL) {
1048878ed226SJulian Elischer 		NG_HCI_ALERT(
1049878ed226SJulian Elischer "%s: %s - invalid link type=%d\n",
1050878ed226SJulian Elischer 			__func__, NG_NODE_NAME(unit->node), con->link_type);
1051878ed226SJulian Elischer 		error = EINVAL;
1052878ed226SJulian Elischer 	} else if (con->state != NG_HCI_CON_OPEN) {
1053878ed226SJulian Elischer 		NG_HCI_ALERT(
1054878ed226SJulian Elischer "%s: %s - invalid connection state=%d, handle=%d\n",
1055878ed226SJulian Elischer 			__func__, NG_NODE_NAME(unit->node), con->state, h);
1056878ed226SJulian Elischer 		error = EINVAL;
1057878ed226SJulian Elischer 	} else /* Notify upper layer */
1058878ed226SJulian Elischer 		error = ng_hci_lp_qos_ind(con);
1059878ed226SJulian Elischer 
1060878ed226SJulian Elischer 	NG_FREE_M(event);
1061878ed226SJulian Elischer 
1062878ed226SJulian Elischer 	return (error);
1063878ed226SJulian Elischer } /* qos_violation */
1064878ed226SJulian Elischer 
1065878ed226SJulian Elischer /* Page scan mode change event */
1066878ed226SJulian Elischer static int
1067878ed226SJulian Elischer page_scan_mode_change(ng_hci_unit_p unit, struct mbuf *event)
1068878ed226SJulian Elischer {
1069878ed226SJulian Elischer 	ng_hci_page_scan_mode_change_ep	*ep = NULL;
1070878ed226SJulian Elischer 	ng_hci_neighbor_p		 n = NULL;
1071878ed226SJulian Elischer 	int				 error = 0;
1072878ed226SJulian Elischer 
1073878ed226SJulian Elischer 	NG_HCI_M_PULLUP(event, sizeof(*ep));
1074878ed226SJulian Elischer 	if (event == NULL)
1075878ed226SJulian Elischer 		return (ENOBUFS);
1076878ed226SJulian Elischer 
1077878ed226SJulian Elischer 	ep = mtod(event, ng_hci_page_scan_mode_change_ep *);
1078878ed226SJulian Elischer 
1079878ed226SJulian Elischer 	/* Update cache entry */
1080878ed226SJulian Elischer 	n = ng_hci_get_neighbor(unit, &ep->bdaddr);
1081878ed226SJulian Elischer 	if (n == NULL) {
1082878ed226SJulian Elischer 		n = ng_hci_new_neighbor(unit);
1083878ed226SJulian Elischer 		if (n == NULL) {
1084878ed226SJulian Elischer 			error = ENOMEM;
1085878ed226SJulian Elischer 			goto out;
1086878ed226SJulian Elischer 		}
1087878ed226SJulian Elischer 
1088878ed226SJulian Elischer 		bcopy(&ep->bdaddr, &n->bdaddr, sizeof(n->bdaddr));
1089878ed226SJulian Elischer 	} else
1090878ed226SJulian Elischer 		getmicrotime(&n->updated);
1091878ed226SJulian Elischer 
1092878ed226SJulian Elischer 	n->page_scan_mode = ep->page_scan_mode;
1093878ed226SJulian Elischer out:
1094878ed226SJulian Elischer 	NG_FREE_M(event);
1095878ed226SJulian Elischer 
1096878ed226SJulian Elischer 	return (error);
1097878ed226SJulian Elischer } /* page_scan_mode_change */
1098878ed226SJulian Elischer 
1099878ed226SJulian Elischer /* Page scan repetition mode change event */
1100878ed226SJulian Elischer static int
1101878ed226SJulian Elischer page_scan_rep_mode_change(ng_hci_unit_p unit, struct mbuf *event)
1102878ed226SJulian Elischer {
1103878ed226SJulian Elischer 	ng_hci_page_scan_rep_mode_change_ep	*ep = NULL;
1104878ed226SJulian Elischer 	ng_hci_neighbor_p			 n = NULL;
1105878ed226SJulian Elischer 	int					 error = 0;
1106878ed226SJulian Elischer 
1107878ed226SJulian Elischer 	NG_HCI_M_PULLUP(event, sizeof(*ep));
1108878ed226SJulian Elischer 	if (event == NULL)
1109878ed226SJulian Elischer 		return (ENOBUFS);
1110878ed226SJulian Elischer 
1111878ed226SJulian Elischer 	ep = mtod(event, ng_hci_page_scan_rep_mode_change_ep *);
1112878ed226SJulian Elischer 
1113878ed226SJulian Elischer 	/* Update cache entry */
1114878ed226SJulian Elischer 	n = ng_hci_get_neighbor(unit, &ep->bdaddr);
1115878ed226SJulian Elischer 	if (n == NULL) {
1116878ed226SJulian Elischer 		n = ng_hci_new_neighbor(unit);
1117878ed226SJulian Elischer 		if (n == NULL) {
1118878ed226SJulian Elischer 			error = ENOMEM;
1119878ed226SJulian Elischer 			goto out;
1120878ed226SJulian Elischer 		}
1121878ed226SJulian Elischer 
1122878ed226SJulian Elischer 		bcopy(&ep->bdaddr, &n->bdaddr, sizeof(n->bdaddr));
1123878ed226SJulian Elischer 	} else
1124878ed226SJulian Elischer 		getmicrotime(&n->updated);
1125878ed226SJulian Elischer 
1126878ed226SJulian Elischer 	n->page_scan_rep_mode = ep->page_scan_rep_mode;
1127878ed226SJulian Elischer out:
1128878ed226SJulian Elischer 	NG_FREE_M(event);
1129878ed226SJulian Elischer 
1130878ed226SJulian Elischer 	return (error);
1131878ed226SJulian Elischer } /* page_scan_rep_mode_change */
1132878ed226SJulian Elischer 
1133