xref: /freebsd/sys/netgraph/bluetooth/hci/ng_hci_cmds.c (revision fbc48c2bfb4b1d5c7901127694cf6653cf1cf924)
1878ed226SJulian Elischer /*
2878ed226SJulian Elischer  * ng_hci_cmds.c
3c398230bSWarner Losh  */
4c398230bSWarner Losh 
5c398230bSWarner Losh /*-
6878ed226SJulian Elischer  * Copyright (c) Maksim Yevmenkin <m_evmenkin@yahoo.com>
7878ed226SJulian Elischer  * All rights reserved.
8878ed226SJulian Elischer  *
9878ed226SJulian Elischer  * Redistribution and use in source and binary forms, with or without
10878ed226SJulian Elischer  * modification, are permitted provided that the following conditions
11878ed226SJulian Elischer  * are met:
12878ed226SJulian Elischer  * 1. Redistributions of source code must retain the above copyright
13878ed226SJulian Elischer  *    notice, this list of conditions and the following disclaimer.
14878ed226SJulian Elischer  * 2. Redistributions in binary form must reproduce the above copyright
15878ed226SJulian Elischer  *    notice, this list of conditions and the following disclaimer in the
16878ed226SJulian Elischer  *    documentation and/or other materials provided with the distribution.
17878ed226SJulian Elischer  *
18878ed226SJulian Elischer  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
19878ed226SJulian Elischer  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
20878ed226SJulian Elischer  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
21878ed226SJulian Elischer  * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
22878ed226SJulian Elischer  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
23878ed226SJulian Elischer  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
24878ed226SJulian Elischer  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
25878ed226SJulian Elischer  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
26878ed226SJulian Elischer  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
27878ed226SJulian Elischer  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
28878ed226SJulian Elischer  * SUCH DAMAGE.
29878ed226SJulian Elischer  *
300986ab12SMaksim Yevmenkin  * $Id: ng_hci_cmds.c,v 1.4 2003/09/08 18:57:51 max Exp $
31878ed226SJulian Elischer  * $FreeBSD$
32878ed226SJulian Elischer  */
33878ed226SJulian Elischer 
34878ed226SJulian Elischer #include <sys/param.h>
35878ed226SJulian Elischer #include <sys/systm.h>
36878ed226SJulian Elischer #include <sys/kernel.h>
37878ed226SJulian Elischer #include <sys/endian.h>
38878ed226SJulian Elischer #include <sys/malloc.h>
39878ed226SJulian Elischer #include <sys/mbuf.h>
40878ed226SJulian Elischer #include <sys/queue.h>
41878ed226SJulian Elischer #include <netgraph/ng_message.h>
42878ed226SJulian Elischer #include <netgraph/netgraph.h>
43b84b10f9SMaksim Yevmenkin #include <netgraph/bluetooth/include/ng_bluetooth.h>
44b84b10f9SMaksim Yevmenkin #include <netgraph/bluetooth/include/ng_hci.h>
45b84b10f9SMaksim Yevmenkin #include <netgraph/bluetooth/hci/ng_hci_var.h>
46b84b10f9SMaksim Yevmenkin #include <netgraph/bluetooth/hci/ng_hci_cmds.h>
47b84b10f9SMaksim Yevmenkin #include <netgraph/bluetooth/hci/ng_hci_evnt.h>
48b84b10f9SMaksim Yevmenkin #include <netgraph/bluetooth/hci/ng_hci_ulpi.h>
49b84b10f9SMaksim Yevmenkin #include <netgraph/bluetooth/hci/ng_hci_misc.h>
50878ed226SJulian Elischer 
51878ed226SJulian Elischer /******************************************************************************
52878ed226SJulian Elischer  ******************************************************************************
53878ed226SJulian Elischer  **                     HCI commands processing module
54878ed226SJulian Elischer  ******************************************************************************
55878ed226SJulian Elischer  ******************************************************************************/
56878ed226SJulian Elischer 
57878ed226SJulian Elischer #undef	min
58878ed226SJulian Elischer #define	min(a, b)	((a) < (b))? (a) : (b)
59878ed226SJulian Elischer 
60878ed226SJulian Elischer static int  complete_command (ng_hci_unit_p, int, struct mbuf **);
61878ed226SJulian Elischer 
62878ed226SJulian Elischer static int process_link_control_params
63878ed226SJulian Elischer 	(ng_hci_unit_p, u_int16_t, struct mbuf *, struct mbuf *);
64878ed226SJulian Elischer static int process_link_policy_params
65878ed226SJulian Elischer 	(ng_hci_unit_p, u_int16_t, struct mbuf *, struct mbuf *);
66878ed226SJulian Elischer static int process_hc_baseband_params
67878ed226SJulian Elischer 	(ng_hci_unit_p, u_int16_t, struct mbuf *, struct mbuf *);
68878ed226SJulian Elischer static int process_info_params
69878ed226SJulian Elischer 	(ng_hci_unit_p, u_int16_t, struct mbuf *, struct mbuf *);
70878ed226SJulian Elischer static int process_status_params
71878ed226SJulian Elischer 	(ng_hci_unit_p, u_int16_t, struct mbuf *, struct mbuf *);
72878ed226SJulian Elischer static int process_testing_params
73878ed226SJulian Elischer 	(ng_hci_unit_p, u_int16_t, struct mbuf *, struct mbuf *);
74*fbc48c2bSTakanori Watanabe static int process_le_params
75*fbc48c2bSTakanori Watanabe 	(ng_hci_unit_p, u_int16_t, struct mbuf *, struct mbuf *);
76878ed226SJulian Elischer 
77878ed226SJulian Elischer static int process_link_control_status
78878ed226SJulian Elischer 	(ng_hci_unit_p, ng_hci_command_status_ep *, struct mbuf *);
79878ed226SJulian Elischer static int process_link_policy_status
80878ed226SJulian Elischer 	(ng_hci_unit_p, ng_hci_command_status_ep *, struct mbuf *);
81*fbc48c2bSTakanori Watanabe static int process_le_status
82*fbc48c2bSTakanori Watanabe 	(ng_hci_unit_p, ng_hci_command_status_ep *, struct mbuf *);
83878ed226SJulian Elischer 
84878ed226SJulian Elischer /*
85878ed226SJulian Elischer  * Send HCI command to the driver.
86878ed226SJulian Elischer  */
87878ed226SJulian Elischer 
88878ed226SJulian Elischer int
89878ed226SJulian Elischer ng_hci_send_command(ng_hci_unit_p unit)
90878ed226SJulian Elischer {
91878ed226SJulian Elischer 	struct mbuf	*m0 = NULL, *m = NULL;
92878ed226SJulian Elischer 	int		 free, error = 0;
93878ed226SJulian Elischer 
94878ed226SJulian Elischer 	/* Check if other command is pending */
95878ed226SJulian Elischer 	if (unit->state & NG_HCI_UNIT_COMMAND_PENDING)
96878ed226SJulian Elischer 		return (0);
97878ed226SJulian Elischer 
98878ed226SJulian Elischer 	/* Check if unit can accept our command */
99878ed226SJulian Elischer 	NG_HCI_BUFF_CMD_GET(unit->buffer, free);
100878ed226SJulian Elischer 	if (free == 0)
101878ed226SJulian Elischer 		return (0);
102878ed226SJulian Elischer 
103878ed226SJulian Elischer 	/* Check if driver hook is still ok */
104878ed226SJulian Elischer 	if (unit->drv == NULL || NG_HOOK_NOT_VALID(unit->drv)) {
105878ed226SJulian Elischer 		NG_HCI_WARN(
106878ed226SJulian Elischer "%s: %s - hook \"%s\" is not connected or valid\n",
107878ed226SJulian Elischer 			__func__, NG_NODE_NAME(unit->node), NG_HCI_HOOK_DRV);
108878ed226SJulian Elischer 
109878ed226SJulian Elischer 		NG_BT_MBUFQ_DRAIN(&unit->cmdq);
110878ed226SJulian Elischer 
111878ed226SJulian Elischer 		return (ENOTCONN);
112878ed226SJulian Elischer 	}
113878ed226SJulian Elischer 
114878ed226SJulian Elischer 	/*
115878ed226SJulian Elischer 	 * Get first command from queue, give it to RAW hook then
116878ed226SJulian Elischer 	 * make copy of it and send it to the driver
117878ed226SJulian Elischer 	 */
118878ed226SJulian Elischer 
119878ed226SJulian Elischer 	m0 = NG_BT_MBUFQ_FIRST(&unit->cmdq);
120878ed226SJulian Elischer 	if (m0 == NULL)
121878ed226SJulian Elischer 		return (0);
122878ed226SJulian Elischer 
123878ed226SJulian Elischer 	ng_hci_mtap(unit, m0);
124878ed226SJulian Elischer 
125eb1b1807SGleb Smirnoff 	m = m_dup(m0, M_NOWAIT);
126878ed226SJulian Elischer 	if (m != NULL)
127878ed226SJulian Elischer 		NG_SEND_DATA_ONLY(error, unit->drv, m);
128878ed226SJulian Elischer 	else
129878ed226SJulian Elischer 		error = ENOBUFS;
130878ed226SJulian Elischer 
131878ed226SJulian Elischer 	if (error != 0)
132878ed226SJulian Elischer 		NG_HCI_ERR(
133878ed226SJulian Elischer "%s: %s - could not send HCI command, error=%d\n",
134878ed226SJulian Elischer 			__func__, NG_NODE_NAME(unit->node), error);
135878ed226SJulian Elischer 
136878ed226SJulian Elischer 	/*
137878ed226SJulian Elischer 	 * Even if we were not able to send command we still pretend
138878ed226SJulian Elischer 	 * that everything is OK and let timeout handle that.
139878ed226SJulian Elischer 	 */
140878ed226SJulian Elischer 
141878ed226SJulian Elischer 	NG_HCI_BUFF_CMD_USE(unit->buffer, 1);
142878ed226SJulian Elischer 	NG_HCI_STAT_CMD_SENT(unit->stat);
143878ed226SJulian Elischer 	NG_HCI_STAT_BYTES_SENT(unit->stat, m0->m_pkthdr.len);
144878ed226SJulian Elischer 
145878ed226SJulian Elischer 	/*
146878ed226SJulian Elischer 	 * Note: ng_hci_command_timeout() will set
147878ed226SJulian Elischer 	 * NG_HCI_UNIT_COMMAND_PENDING flag
148878ed226SJulian Elischer 	 */
149878ed226SJulian Elischer 
150878ed226SJulian Elischer 	ng_hci_command_timeout(unit);
151878ed226SJulian Elischer 
152878ed226SJulian Elischer 	return (0);
153878ed226SJulian Elischer } /* ng_hci_send_command */
154878ed226SJulian Elischer 
155878ed226SJulian Elischer /*
156878ed226SJulian Elischer  * Process HCI Command_Compete event. Complete HCI command, and do post
157878ed226SJulian Elischer  * processing on the command parameters (cp) and command return parameters
158878ed226SJulian Elischer  * (e) if required (for example adjust state).
159878ed226SJulian Elischer  */
160878ed226SJulian Elischer 
161878ed226SJulian Elischer int
162878ed226SJulian Elischer ng_hci_process_command_complete(ng_hci_unit_p unit, struct mbuf *e)
163878ed226SJulian Elischer {
164878ed226SJulian Elischer 	ng_hci_command_compl_ep		*ep = NULL;
165878ed226SJulian Elischer 	struct mbuf			*cp = NULL;
166878ed226SJulian Elischer 	int				 error = 0;
167878ed226SJulian Elischer 
168878ed226SJulian Elischer 	/* Get event packet and update command buffer info */
169878ed226SJulian Elischer 	NG_HCI_M_PULLUP(e, sizeof(*ep));
170878ed226SJulian Elischer 	if (e == NULL)
171878ed226SJulian Elischer 		return (ENOBUFS); /* XXX this is bad */
172878ed226SJulian Elischer 
173878ed226SJulian Elischer 	ep = mtod(e, ng_hci_command_compl_ep *);
174878ed226SJulian Elischer         NG_HCI_BUFF_CMD_SET(unit->buffer, ep->num_cmd_pkts);
175878ed226SJulian Elischer 
176878ed226SJulian Elischer 	/* Check for special NOOP command */
177878ed226SJulian Elischer 	if (ep->opcode == 0x0000) {
178878ed226SJulian Elischer 		NG_FREE_M(e);
179878ed226SJulian Elischer 		goto out;
180878ed226SJulian Elischer 	}
181878ed226SJulian Elischer 
182878ed226SJulian Elischer 	/* Try to match first command item in the queue */
183878ed226SJulian Elischer 	error = complete_command(unit, ep->opcode, &cp);
184878ed226SJulian Elischer 	if (error != 0) {
185878ed226SJulian Elischer 		NG_FREE_M(e);
186878ed226SJulian Elischer 		goto out;
187878ed226SJulian Elischer 	}
188878ed226SJulian Elischer 
189878ed226SJulian Elischer 	/*
190878ed226SJulian Elischer 	 * Perform post processing on command parameters and return parameters
191878ed226SJulian Elischer 	 * do it only if status is OK (status == 0). Status is the first byte
192878ed226SJulian Elischer 	 * of any command return parameters.
193878ed226SJulian Elischer 	 */
194878ed226SJulian Elischer 
195878ed226SJulian Elischer 	ep->opcode = le16toh(ep->opcode);
196878ed226SJulian Elischer 	m_adj(e, sizeof(*ep));
197878ed226SJulian Elischer 
198878ed226SJulian Elischer 	if (*mtod(e, u_int8_t *) == 0) { /* XXX m_pullup here? */
199878ed226SJulian Elischer 		switch (NG_HCI_OGF(ep->opcode)) {
200878ed226SJulian Elischer 		case NG_HCI_OGF_LINK_CONTROL:
201878ed226SJulian Elischer 			error = process_link_control_params(unit,
202878ed226SJulian Elischer 					NG_HCI_OCF(ep->opcode), cp, e);
203878ed226SJulian Elischer 			break;
204878ed226SJulian Elischer 
205878ed226SJulian Elischer 		case NG_HCI_OGF_LINK_POLICY:
206878ed226SJulian Elischer 			error = process_link_policy_params(unit,
207878ed226SJulian Elischer 					NG_HCI_OCF(ep->opcode), cp, e);
208878ed226SJulian Elischer 			break;
209878ed226SJulian Elischer 
210878ed226SJulian Elischer 		case NG_HCI_OGF_HC_BASEBAND:
211878ed226SJulian Elischer 			error = process_hc_baseband_params(unit,
212878ed226SJulian Elischer 					NG_HCI_OCF(ep->opcode), cp, e);
213878ed226SJulian Elischer 			break;
214878ed226SJulian Elischer 
215878ed226SJulian Elischer 		case NG_HCI_OGF_INFO:
216878ed226SJulian Elischer 			error = process_info_params(unit,
217878ed226SJulian Elischer 					NG_HCI_OCF(ep->opcode), cp, e);
218878ed226SJulian Elischer 			break;
219878ed226SJulian Elischer 
220878ed226SJulian Elischer 		case NG_HCI_OGF_STATUS:
221878ed226SJulian Elischer 			error = process_status_params(unit,
222878ed226SJulian Elischer 					NG_HCI_OCF(ep->opcode), cp, e);
223878ed226SJulian Elischer 			break;
224878ed226SJulian Elischer 
225878ed226SJulian Elischer 		case NG_HCI_OGF_TESTING:
226878ed226SJulian Elischer 			error = process_testing_params(unit,
227878ed226SJulian Elischer 					NG_HCI_OCF(ep->opcode), cp, e);
228878ed226SJulian Elischer 			break;
229*fbc48c2bSTakanori Watanabe 		case NG_HCI_OGF_LE:
230*fbc48c2bSTakanori Watanabe 			error = process_le_params(unit,
231*fbc48c2bSTakanori Watanabe 					  NG_HCI_OCF(ep->opcode), cp, e);
232*fbc48c2bSTakanori Watanabe 			break;
233878ed226SJulian Elischer 		case NG_HCI_OGF_BT_LOGO:
234878ed226SJulian Elischer 		case NG_HCI_OGF_VENDOR:
235878ed226SJulian Elischer 			NG_FREE_M(cp);
236878ed226SJulian Elischer 			NG_FREE_M(e);
237878ed226SJulian Elischer 			break;
238878ed226SJulian Elischer 
239878ed226SJulian Elischer 		default:
240878ed226SJulian Elischer 			NG_FREE_M(cp);
241878ed226SJulian Elischer 			NG_FREE_M(e);
242878ed226SJulian Elischer 			error = EINVAL;
243878ed226SJulian Elischer 			break;
244878ed226SJulian Elischer 		}
245878ed226SJulian Elischer 	} else {
246878ed226SJulian Elischer 		NG_HCI_ERR(
247878ed226SJulian Elischer "%s: %s - HCI command failed, OGF=%#x, OCF=%#x, status=%#x\n",
248878ed226SJulian Elischer 			__func__, NG_NODE_NAME(unit->node),
249878ed226SJulian Elischer 			NG_HCI_OGF(ep->opcode), NG_HCI_OCF(ep->opcode),
250878ed226SJulian Elischer 			*mtod(e, u_int8_t *));
251878ed226SJulian Elischer 
252878ed226SJulian Elischer 		NG_FREE_M(cp);
253878ed226SJulian Elischer 		NG_FREE_M(e);
254878ed226SJulian Elischer 	}
255878ed226SJulian Elischer out:
256878ed226SJulian Elischer 	ng_hci_send_command(unit);
257878ed226SJulian Elischer 
258878ed226SJulian Elischer 	return (error);
259878ed226SJulian Elischer } /* ng_hci_process_command_complete */
260878ed226SJulian Elischer 
261878ed226SJulian Elischer /*
262878ed226SJulian Elischer  * Process HCI Command_Status event. Check the status (mst) and do post
263878ed226SJulian Elischer  * processing (if required).
264878ed226SJulian Elischer  */
265878ed226SJulian Elischer 
266878ed226SJulian Elischer int
267878ed226SJulian Elischer ng_hci_process_command_status(ng_hci_unit_p unit, struct mbuf *e)
268878ed226SJulian Elischer {
269878ed226SJulian Elischer 	ng_hci_command_status_ep	*ep = NULL;
270878ed226SJulian Elischer 	struct mbuf			*cp = NULL;
271878ed226SJulian Elischer 	int				 error = 0;
272878ed226SJulian Elischer 
273878ed226SJulian Elischer 	/* Update command buffer info */
274878ed226SJulian Elischer 	NG_HCI_M_PULLUP(e, sizeof(*ep));
275878ed226SJulian Elischer 	if (e == NULL)
276878ed226SJulian Elischer 		return (ENOBUFS); /* XXX this is bad */
277878ed226SJulian Elischer 
278878ed226SJulian Elischer 	ep = mtod(e, ng_hci_command_status_ep *);
279878ed226SJulian Elischer 	NG_HCI_BUFF_CMD_SET(unit->buffer, ep->num_cmd_pkts);
280878ed226SJulian Elischer 
281878ed226SJulian Elischer 	/* Check for special NOOP command */
282878ed226SJulian Elischer 	if (ep->opcode == 0x0000)
283878ed226SJulian Elischer 		goto out;
284878ed226SJulian Elischer 
285878ed226SJulian Elischer 	/* Try to match first command item in the queue */
286878ed226SJulian Elischer 	error = complete_command(unit, ep->opcode, &cp);
287878ed226SJulian Elischer         if (error != 0)
288878ed226SJulian Elischer 		goto out;
289878ed226SJulian Elischer 
290878ed226SJulian Elischer 	/*
291878ed226SJulian Elischer 	 * Perform post processing on HCI Command_Status event
292878ed226SJulian Elischer 	 */
293878ed226SJulian Elischer 
294878ed226SJulian Elischer 	ep->opcode = le16toh(ep->opcode);
295878ed226SJulian Elischer 
296878ed226SJulian Elischer 	switch (NG_HCI_OGF(ep->opcode)) {
297878ed226SJulian Elischer 	case NG_HCI_OGF_LINK_CONTROL:
298878ed226SJulian Elischer 		error = process_link_control_status(unit, ep, cp);
299878ed226SJulian Elischer 		break;
300878ed226SJulian Elischer 
301878ed226SJulian Elischer 	case NG_HCI_OGF_LINK_POLICY:
302878ed226SJulian Elischer 		error = process_link_policy_status(unit, ep, cp);
303878ed226SJulian Elischer 		break;
304*fbc48c2bSTakanori Watanabe 	case NG_HCI_OGF_LE:
305*fbc48c2bSTakanori Watanabe 		error = process_le_status(unit, ep, cp);
306*fbc48c2bSTakanori Watanabe 		break;
307878ed226SJulian Elischer 	case NG_HCI_OGF_BT_LOGO:
308878ed226SJulian Elischer 	case NG_HCI_OGF_VENDOR:
309878ed226SJulian Elischer 		NG_FREE_M(cp);
310878ed226SJulian Elischer 		break;
311878ed226SJulian Elischer 
312878ed226SJulian Elischer 	case NG_HCI_OGF_HC_BASEBAND:
313878ed226SJulian Elischer 	case NG_HCI_OGF_INFO:
314878ed226SJulian Elischer 	case NG_HCI_OGF_STATUS:
315878ed226SJulian Elischer 	case NG_HCI_OGF_TESTING:
316878ed226SJulian Elischer 	default:
317878ed226SJulian Elischer 		NG_FREE_M(cp);
318878ed226SJulian Elischer 		error = EINVAL;
319878ed226SJulian Elischer 		break;
320878ed226SJulian Elischer 	}
321878ed226SJulian Elischer out:
322878ed226SJulian Elischer 	NG_FREE_M(e);
323878ed226SJulian Elischer 	ng_hci_send_command(unit);
324878ed226SJulian Elischer 
325878ed226SJulian Elischer 	return (error);
326878ed226SJulian Elischer } /* ng_hci_process_command_status */
327878ed226SJulian Elischer 
328878ed226SJulian Elischer /*
329878ed226SJulian Elischer  * Complete queued HCI command.
330878ed226SJulian Elischer  */
331878ed226SJulian Elischer 
332878ed226SJulian Elischer static int
333878ed226SJulian Elischer complete_command(ng_hci_unit_p unit, int opcode, struct mbuf **cp)
334878ed226SJulian Elischer {
335878ed226SJulian Elischer 	struct mbuf	*m = NULL;
336878ed226SJulian Elischer 
337878ed226SJulian Elischer 	/* Check unit state */
338878ed226SJulian Elischer 	if (!(unit->state & NG_HCI_UNIT_COMMAND_PENDING)) {
339878ed226SJulian Elischer 		NG_HCI_ALERT(
340878ed226SJulian Elischer "%s: %s - no pending command, state=%#x\n",
341878ed226SJulian Elischer 			__func__, NG_NODE_NAME(unit->node), unit->state);
342878ed226SJulian Elischer 
343878ed226SJulian Elischer 		return (EINVAL);
344878ed226SJulian Elischer 	}
345878ed226SJulian Elischer 
346878ed226SJulian Elischer 	/* Get first command in the queue */
347878ed226SJulian Elischer 	m = NG_BT_MBUFQ_FIRST(&unit->cmdq);
348878ed226SJulian Elischer 	if (m == NULL) {
349878ed226SJulian Elischer 		NG_HCI_ALERT(
350878ed226SJulian Elischer "%s: %s - empty command queue?!\n", __func__, NG_NODE_NAME(unit->node));
351878ed226SJulian Elischer 
352878ed226SJulian Elischer 		return (EINVAL);
353878ed226SJulian Elischer 	}
354878ed226SJulian Elischer 
355878ed226SJulian Elischer 	/*
356878ed226SJulian Elischer 	 * Match command opcode, if does not match - do nothing and
357878ed226SJulian Elischer 	 * let timeout handle that.
358878ed226SJulian Elischer 	 */
359878ed226SJulian Elischer 
360878ed226SJulian Elischer 	if (mtod(m, ng_hci_cmd_pkt_t *)->opcode != opcode) {
361878ed226SJulian Elischer 		NG_HCI_ALERT(
362878ed226SJulian Elischer "%s: %s - command queue is out of sync\n", __func__, NG_NODE_NAME(unit->node));
363878ed226SJulian Elischer 
364878ed226SJulian Elischer 		return (EINVAL);
365878ed226SJulian Elischer 	}
366878ed226SJulian Elischer 
367878ed226SJulian Elischer 	/*
368878ed226SJulian Elischer 	 * Now we can remove command timeout, dequeue completed command
3690986ab12SMaksim Yevmenkin 	 * and return command parameters. ng_hci_command_untimeout will
3700986ab12SMaksim Yevmenkin 	 * drop NG_HCI_UNIT_COMMAND_PENDING flag.
3710986ab12SMaksim Yevmenkin 	 * Note: if ng_hci_command_untimeout() fails (returns non-zero)
3720986ab12SMaksim Yevmenkin 	 * then timeout aready happened and timeout message went info node
3730986ab12SMaksim Yevmenkin 	 * queue. In this case we ignore command completion and pretend
3740986ab12SMaksim Yevmenkin 	 * there is a timeout.
375878ed226SJulian Elischer 	 */
376878ed226SJulian Elischer 
3770986ab12SMaksim Yevmenkin 	if (ng_hci_command_untimeout(unit) != 0)
3780986ab12SMaksim Yevmenkin 		return (ETIMEDOUT);
3790986ab12SMaksim Yevmenkin 
380878ed226SJulian Elischer 	NG_BT_MBUFQ_DEQUEUE(&unit->cmdq, *cp);
381878ed226SJulian Elischer 	m_adj(*cp, sizeof(ng_hci_cmd_pkt_t));
382878ed226SJulian Elischer 
383878ed226SJulian Elischer 	return (0);
384878ed226SJulian Elischer } /* complete_command */
385878ed226SJulian Elischer 
386878ed226SJulian Elischer /*
387878ed226SJulian Elischer  * Process HCI command timeout
388878ed226SJulian Elischer  */
389878ed226SJulian Elischer 
390878ed226SJulian Elischer void
391878ed226SJulian Elischer ng_hci_process_command_timeout(node_p node, hook_p hook, void *arg1, int arg2)
392878ed226SJulian Elischer {
3930986ab12SMaksim Yevmenkin 	ng_hci_unit_p	 unit = NULL;
394878ed226SJulian Elischer 	struct mbuf	*m = NULL;
395878ed226SJulian Elischer 	u_int16_t	 opcode;
396878ed226SJulian Elischer 
3970986ab12SMaksim Yevmenkin 	if (NG_NODE_NOT_VALID(node)) {
3980986ab12SMaksim Yevmenkin 		printf("%s: Netgraph node is not valid\n", __func__);
3990986ab12SMaksim Yevmenkin 		return;
4000986ab12SMaksim Yevmenkin 	}
401878ed226SJulian Elischer 
4020986ab12SMaksim Yevmenkin 	unit = (ng_hci_unit_p) NG_NODE_PRIVATE(node);
4030986ab12SMaksim Yevmenkin 
4040986ab12SMaksim Yevmenkin 	if (unit->state & NG_HCI_UNIT_COMMAND_PENDING) {
4050986ab12SMaksim Yevmenkin 		unit->state &= ~NG_HCI_UNIT_COMMAND_PENDING;
4060986ab12SMaksim Yevmenkin 
4070986ab12SMaksim Yevmenkin 		NG_BT_MBUFQ_DEQUEUE(&unit->cmdq, m);
408878ed226SJulian Elischer 		if (m == NULL) {
4090986ab12SMaksim Yevmenkin 			NG_HCI_ALERT(
4100986ab12SMaksim Yevmenkin "%s: %s - command queue is out of sync!\n", __func__, NG_NODE_NAME(unit->node));
411878ed226SJulian Elischer 
412878ed226SJulian Elischer 			return;
413878ed226SJulian Elischer 		}
414878ed226SJulian Elischer 
415878ed226SJulian Elischer 		opcode = le16toh(mtod(m, ng_hci_cmd_pkt_t *)->opcode);
416878ed226SJulian Elischer 		NG_FREE_M(m);
417878ed226SJulian Elischer 
418878ed226SJulian Elischer 		NG_HCI_ERR(
419878ed226SJulian Elischer "%s: %s - unable to complete HCI command OGF=%#x, OCF=%#x. Timeout\n",
420878ed226SJulian Elischer 			__func__, NG_NODE_NAME(unit->node), NG_HCI_OGF(opcode),
421878ed226SJulian Elischer 			NG_HCI_OCF(opcode));
422878ed226SJulian Elischer 
4230986ab12SMaksim Yevmenkin 		/* Try to send more commands */
424878ed226SJulian Elischer  		NG_HCI_BUFF_CMD_SET(unit->buffer, 1);
425878ed226SJulian Elischer 		ng_hci_send_command(unit);
426878ed226SJulian Elischer 	} else
4270986ab12SMaksim Yevmenkin 		NG_HCI_ALERT(
4280986ab12SMaksim Yevmenkin "%s: %s - no pending command\n", __func__, NG_NODE_NAME(unit->node));
429878ed226SJulian Elischer } /* ng_hci_process_command_timeout */
430878ed226SJulian Elischer 
431878ed226SJulian Elischer /*
432878ed226SJulian Elischer  * Process link command return parameters
433878ed226SJulian Elischer  */
434878ed226SJulian Elischer 
435878ed226SJulian Elischer static int
436878ed226SJulian Elischer process_link_control_params(ng_hci_unit_p unit, u_int16_t ocf,
437878ed226SJulian Elischer 		struct mbuf *mcp, struct mbuf *mrp)
438878ed226SJulian Elischer {
439878ed226SJulian Elischer 	int	error  = 0;
440878ed226SJulian Elischer 
441878ed226SJulian Elischer 	switch (ocf) {
442878ed226SJulian Elischer 	case NG_HCI_OCF_INQUIRY_CANCEL:
443878ed226SJulian Elischer 	case NG_HCI_OCF_PERIODIC_INQUIRY:
444878ed226SJulian Elischer 	case NG_HCI_OCF_EXIT_PERIODIC_INQUIRY:
445878ed226SJulian Elischer 	case NG_HCI_OCF_LINK_KEY_REP:
446878ed226SJulian Elischer 	case NG_HCI_OCF_LINK_KEY_NEG_REP:
447878ed226SJulian Elischer 	case NG_HCI_OCF_PIN_CODE_REP:
448878ed226SJulian Elischer 	case NG_HCI_OCF_PIN_CODE_NEG_REP:
449878ed226SJulian Elischer 		/* These do not need post processing */
450878ed226SJulian Elischer 		break;
451878ed226SJulian Elischer 
452878ed226SJulian Elischer 	case NG_HCI_OCF_INQUIRY:
453878ed226SJulian Elischer 	case NG_HCI_OCF_CREATE_CON:
454878ed226SJulian Elischer 	case NG_HCI_OCF_DISCON:
455878ed226SJulian Elischer 	case NG_HCI_OCF_ADD_SCO_CON:
456878ed226SJulian Elischer 	case NG_HCI_OCF_ACCEPT_CON:
457878ed226SJulian Elischer 	case NG_HCI_OCF_REJECT_CON:
458878ed226SJulian Elischer 	case NG_HCI_OCF_CHANGE_CON_PKT_TYPE:
459878ed226SJulian Elischer 	case NG_HCI_OCF_AUTH_REQ:
460878ed226SJulian Elischer 	case NG_HCI_OCF_SET_CON_ENCRYPTION:
461878ed226SJulian Elischer 	case NG_HCI_OCF_CHANGE_CON_LINK_KEY:
462878ed226SJulian Elischer 	case NG_HCI_OCF_MASTER_LINK_KEY:
463878ed226SJulian Elischer 	case NG_HCI_OCF_REMOTE_NAME_REQ:
464878ed226SJulian Elischer 	case NG_HCI_OCF_READ_REMOTE_FEATURES:
465878ed226SJulian Elischer 	case NG_HCI_OCF_READ_REMOTE_VER_INFO:
466878ed226SJulian Elischer 	case NG_HCI_OCF_READ_CLOCK_OFFSET:
467878ed226SJulian Elischer 	default:
468878ed226SJulian Elischer 
469878ed226SJulian Elischer 		/*
470878ed226SJulian Elischer 		 * None of these command was supposed to generate
471878ed226SJulian Elischer 		 * Command_Complete event. Instead Command_Status event
472878ed226SJulian Elischer 		 * should have been generated and then appropriate event
473878ed226SJulian Elischer 		 * should have been sent to indicate the final result.
474878ed226SJulian Elischer 		 */
475878ed226SJulian Elischer 
476878ed226SJulian Elischer 		error = EINVAL;
477878ed226SJulian Elischer 		break;
478878ed226SJulian Elischer 	}
479878ed226SJulian Elischer 
480878ed226SJulian Elischer 	NG_FREE_M(mcp);
481878ed226SJulian Elischer 	NG_FREE_M(mrp);
482878ed226SJulian Elischer 
483878ed226SJulian Elischer 	return (error);
484878ed226SJulian Elischer } /* process_link_control_params */
485878ed226SJulian Elischer 
486878ed226SJulian Elischer /*
487878ed226SJulian Elischer  * Process link policy command return parameters
488878ed226SJulian Elischer  */
489878ed226SJulian Elischer 
490878ed226SJulian Elischer static int
491878ed226SJulian Elischer process_link_policy_params(ng_hci_unit_p unit, u_int16_t ocf,
492878ed226SJulian Elischer 		struct mbuf *mcp, struct mbuf *mrp)
493878ed226SJulian Elischer {
494878ed226SJulian Elischer 	int	error = 0;
495878ed226SJulian Elischer 
496878ed226SJulian Elischer 	switch (ocf){
497878ed226SJulian Elischer 	case NG_HCI_OCF_ROLE_DISCOVERY: {
498878ed226SJulian Elischer 		ng_hci_role_discovery_rp	*rp = NULL;
499878ed226SJulian Elischer 		ng_hci_unit_con_t		*con = NULL;
500878ed226SJulian Elischer 		u_int16_t			 h;
501878ed226SJulian Elischer 
502878ed226SJulian Elischer 		NG_HCI_M_PULLUP(mrp, sizeof(*rp));
503878ed226SJulian Elischer 		if (mrp != NULL) {
504878ed226SJulian Elischer 			rp = mtod(mrp, ng_hci_role_discovery_rp *);
505878ed226SJulian Elischer 
506878ed226SJulian Elischer 			h = NG_HCI_CON_HANDLE(le16toh(rp->con_handle));
507878ed226SJulian Elischer 			con = ng_hci_con_by_handle(unit, h);
508878ed226SJulian Elischer 			if (con == NULL) {
509878ed226SJulian Elischer 				NG_HCI_ALERT(
510878ed226SJulian Elischer "%s: %s - invalid connection handle=%d\n",
511878ed226SJulian Elischer 					__func__, NG_NODE_NAME(unit->node), h);
512878ed226SJulian Elischer 				error = ENOENT;
513878ed226SJulian Elischer 			} else if (con->link_type != NG_HCI_LINK_ACL) {
514878ed226SJulian Elischer 				NG_HCI_ALERT(
515878ed226SJulian Elischer "%s: %s - invalid link type=%d\n", __func__, NG_NODE_NAME(unit->node),
516878ed226SJulian Elischer 					con->link_type);
517878ed226SJulian Elischer 				error = EINVAL;
518878ed226SJulian Elischer 			} else
519878ed226SJulian Elischer 				con->role = rp->role;
520878ed226SJulian Elischer 		} else
521878ed226SJulian Elischer 			error = ENOBUFS;
522878ed226SJulian Elischer 		} break;
523878ed226SJulian Elischer 
524878ed226SJulian Elischer 	case NG_HCI_OCF_READ_LINK_POLICY_SETTINGS:
525878ed226SJulian Elischer 	case NG_HCI_OCF_WRITE_LINK_POLICY_SETTINGS:
526878ed226SJulian Elischer 		/* These do not need post processing */
527878ed226SJulian Elischer 		break;
528878ed226SJulian Elischer 
529878ed226SJulian Elischer 	case NG_HCI_OCF_HOLD_MODE:
530878ed226SJulian Elischer 	case NG_HCI_OCF_SNIFF_MODE:
531878ed226SJulian Elischer 	case NG_HCI_OCF_EXIT_SNIFF_MODE:
532878ed226SJulian Elischer 	case NG_HCI_OCF_PARK_MODE:
533878ed226SJulian Elischer 	case NG_HCI_OCF_EXIT_PARK_MODE:
534878ed226SJulian Elischer 	case NG_HCI_OCF_QOS_SETUP:
535878ed226SJulian Elischer 	case NG_HCI_OCF_SWITCH_ROLE:
536878ed226SJulian Elischer 	default:
537878ed226SJulian Elischer 
538878ed226SJulian Elischer 		/*
539878ed226SJulian Elischer 		 * None of these command was supposed to generate
540878ed226SJulian Elischer 		 * Command_Complete event. Instead Command_Status event
541878ed226SJulian Elischer 		 * should have been generated and then appropriate event
542878ed226SJulian Elischer 		 * should have been sent to indicate the final result.
543878ed226SJulian Elischer 		 */
544878ed226SJulian Elischer 
545878ed226SJulian Elischer 		error = EINVAL;
546878ed226SJulian Elischer 		break;
547878ed226SJulian Elischer 	}
548878ed226SJulian Elischer 
549878ed226SJulian Elischer 	NG_FREE_M(mcp);
550878ed226SJulian Elischer 	NG_FREE_M(mrp);
551878ed226SJulian Elischer 
552878ed226SJulian Elischer 	return (error);
553878ed226SJulian Elischer } /* process_link_policy_params */
554878ed226SJulian Elischer 
555878ed226SJulian Elischer /*
556878ed226SJulian Elischer  * Process HC and baseband command return parameters
557878ed226SJulian Elischer  */
558878ed226SJulian Elischer 
559878ed226SJulian Elischer int
560878ed226SJulian Elischer process_hc_baseband_params(ng_hci_unit_p unit, u_int16_t ocf,
561878ed226SJulian Elischer 		struct mbuf *mcp, struct mbuf *mrp)
562878ed226SJulian Elischer {
563878ed226SJulian Elischer 	int	error = 0;
564878ed226SJulian Elischer 
565878ed226SJulian Elischer 	switch (ocf) {
566878ed226SJulian Elischer 	case NG_HCI_OCF_SET_EVENT_MASK:
567878ed226SJulian Elischer 	case NG_HCI_OCF_SET_EVENT_FILTER:
568878ed226SJulian Elischer 	case NG_HCI_OCF_FLUSH:	/* XXX Do we need to handle that? */
569878ed226SJulian Elischer 	case NG_HCI_OCF_READ_PIN_TYPE:
570878ed226SJulian Elischer 	case NG_HCI_OCF_WRITE_PIN_TYPE:
571878ed226SJulian Elischer 	case NG_HCI_OCF_CREATE_NEW_UNIT_KEY:
572878ed226SJulian Elischer 	case NG_HCI_OCF_WRITE_STORED_LINK_KEY:
573878ed226SJulian Elischer 	case NG_HCI_OCF_WRITE_CON_ACCEPT_TIMO:
574878ed226SJulian Elischer 	case NG_HCI_OCF_WRITE_PAGE_TIMO:
575878ed226SJulian Elischer 	case NG_HCI_OCF_READ_SCAN_ENABLE:
576878ed226SJulian Elischer 	case NG_HCI_OCF_WRITE_SCAN_ENABLE:
577878ed226SJulian Elischer 	case NG_HCI_OCF_WRITE_PAGE_SCAN_ACTIVITY:
578878ed226SJulian Elischer 	case NG_HCI_OCF_WRITE_INQUIRY_SCAN_ACTIVITY:
579878ed226SJulian Elischer 	case NG_HCI_OCF_READ_AUTH_ENABLE:
580878ed226SJulian Elischer 	case NG_HCI_OCF_WRITE_AUTH_ENABLE:
581878ed226SJulian Elischer 	case NG_HCI_OCF_READ_ENCRYPTION_MODE:
582878ed226SJulian Elischer 	case NG_HCI_OCF_WRITE_ENCRYPTION_MODE:
583878ed226SJulian Elischer 	case NG_HCI_OCF_WRITE_VOICE_SETTINGS:
584878ed226SJulian Elischer 	case NG_HCI_OCF_READ_NUM_BROADCAST_RETRANS:
585878ed226SJulian Elischer 	case NG_HCI_OCF_WRITE_NUM_BROADCAST_RETRANS:
586878ed226SJulian Elischer 	case NG_HCI_OCF_READ_HOLD_MODE_ACTIVITY:
587878ed226SJulian Elischer 	case NG_HCI_OCF_WRITE_HOLD_MODE_ACTIVITY:
588878ed226SJulian Elischer 	case NG_HCI_OCF_READ_SCO_FLOW_CONTROL:
589878ed226SJulian Elischer 	case NG_HCI_OCF_WRITE_SCO_FLOW_CONTROL:
590878ed226SJulian Elischer 	case NG_HCI_OCF_H2HC_FLOW_CONTROL: /* XXX Not supported this time */
591878ed226SJulian Elischer 	case NG_HCI_OCF_HOST_BUFFER_SIZE:
592878ed226SJulian Elischer 	case NG_HCI_OCF_READ_IAC_LAP:
593878ed226SJulian Elischer 	case NG_HCI_OCF_WRITE_IAC_LAP:
594878ed226SJulian Elischer 	case NG_HCI_OCF_READ_PAGE_SCAN_PERIOD:
595878ed226SJulian Elischer 	case NG_HCI_OCF_WRITE_PAGE_SCAN_PERIOD:
596878ed226SJulian Elischer 	case NG_HCI_OCF_READ_PAGE_SCAN:
597878ed226SJulian Elischer 	case NG_HCI_OCF_WRITE_PAGE_SCAN:
598878ed226SJulian Elischer 	case NG_HCI_OCF_READ_LINK_SUPERVISION_TIMO:
599878ed226SJulian Elischer 	case NG_HCI_OCF_WRITE_LINK_SUPERVISION_TIMO:
600878ed226SJulian Elischer 	case NG_HCI_OCF_READ_SUPPORTED_IAC_NUM:
601878ed226SJulian Elischer 	case NG_HCI_OCF_READ_STORED_LINK_KEY:
602878ed226SJulian Elischer 	case NG_HCI_OCF_DELETE_STORED_LINK_KEY:
603878ed226SJulian Elischer 	case NG_HCI_OCF_READ_CON_ACCEPT_TIMO:
604878ed226SJulian Elischer 	case NG_HCI_OCF_READ_PAGE_TIMO:
605878ed226SJulian Elischer 	case NG_HCI_OCF_READ_PAGE_SCAN_ACTIVITY:
606878ed226SJulian Elischer 	case NG_HCI_OCF_READ_INQUIRY_SCAN_ACTIVITY:
607878ed226SJulian Elischer 	case NG_HCI_OCF_READ_VOICE_SETTINGS:
608878ed226SJulian Elischer 	case NG_HCI_OCF_READ_AUTO_FLUSH_TIMO:
609878ed226SJulian Elischer 	case NG_HCI_OCF_WRITE_AUTO_FLUSH_TIMO:
610878ed226SJulian Elischer 	case NG_HCI_OCF_READ_XMIT_LEVEL:
611878ed226SJulian Elischer 	case NG_HCI_OCF_HOST_NUM_COMPL_PKTS:	/* XXX Can get here? */
612878ed226SJulian Elischer 	case NG_HCI_OCF_CHANGE_LOCAL_NAME:
613878ed226SJulian Elischer 	case NG_HCI_OCF_READ_LOCAL_NAME:
614878ed226SJulian Elischer 	case NG_HCI_OCF_READ_UNIT_CLASS:
615878ed226SJulian Elischer 	case NG_HCI_OCF_WRITE_UNIT_CLASS:
616*fbc48c2bSTakanori Watanabe 	case NG_HCI_OCF_READ_LE_HOST_SUPPORTED:
617*fbc48c2bSTakanori Watanabe 	case NG_HCI_OCF_WRITE_LE_HOST_SUPPORTED:
618878ed226SJulian Elischer 		/* These do not need post processing */
619878ed226SJulian Elischer 		break;
620878ed226SJulian Elischer 
621878ed226SJulian Elischer 	case NG_HCI_OCF_RESET: {
622878ed226SJulian Elischer 		ng_hci_unit_con_p	con = NULL;
623878ed226SJulian Elischer 		int			size;
624878ed226SJulian Elischer 
625878ed226SJulian Elischer 		/*
626878ed226SJulian Elischer 		 * XXX
627878ed226SJulian Elischer 		 *
628878ed226SJulian Elischer 		 * After RESET command unit goes into standby mode
629878ed226SJulian Elischer 		 * and all operational state is lost. Host controller
630878ed226SJulian Elischer 		 * will revert to default values for all parameters.
631878ed226SJulian Elischer 		 *
632878ed226SJulian Elischer 		 * For now we shall terminate all connections and drop
633878ed226SJulian Elischer 		 * inited bit. After RESET unit must be re-initialized.
634878ed226SJulian Elischer 		 */
635878ed226SJulian Elischer 
636878ed226SJulian Elischer 		while (!LIST_EMPTY(&unit->con_list)) {
637878ed226SJulian Elischer 			con = LIST_FIRST(&unit->con_list);
638878ed226SJulian Elischer 
6390986ab12SMaksim Yevmenkin 			/* Remove all timeouts (if any) */
6400986ab12SMaksim Yevmenkin 			if (con->flags & NG_HCI_CON_TIMEOUT_PENDING)
6410986ab12SMaksim Yevmenkin 				ng_hci_con_untimeout(con);
6420986ab12SMaksim Yevmenkin 
643878ed226SJulian Elischer 			/* Connection terminated by local host */
644878ed226SJulian Elischer 			ng_hci_lp_discon_ind(con, 0x16);
645878ed226SJulian Elischer 			ng_hci_free_con(con);
646878ed226SJulian Elischer 		}
647878ed226SJulian Elischer 
648878ed226SJulian Elischer 		NG_HCI_BUFF_ACL_TOTAL(unit->buffer, size);
649878ed226SJulian Elischer 		NG_HCI_BUFF_ACL_FREE(unit->buffer, size);
650878ed226SJulian Elischer 
651878ed226SJulian Elischer 		NG_HCI_BUFF_SCO_TOTAL(unit->buffer, size);
652878ed226SJulian Elischer 		NG_HCI_BUFF_SCO_FREE(unit->buffer, size);
653878ed226SJulian Elischer 
654878ed226SJulian Elischer 		unit->state &= ~NG_HCI_UNIT_INITED;
655878ed226SJulian Elischer 		} break;
656878ed226SJulian Elischer 
657878ed226SJulian Elischer 	default:
658878ed226SJulian Elischer 		error = EINVAL;
659878ed226SJulian Elischer 		break;
660878ed226SJulian Elischer 	}
661878ed226SJulian Elischer 
662878ed226SJulian Elischer 	NG_FREE_M(mcp);
663878ed226SJulian Elischer 	NG_FREE_M(mrp);
664878ed226SJulian Elischer 
665878ed226SJulian Elischer 	return (error);
666878ed226SJulian Elischer } /* process_hc_baseband_params */
667878ed226SJulian Elischer 
668878ed226SJulian Elischer /*
669878ed226SJulian Elischer  * Process info command return parameters
670878ed226SJulian Elischer  */
671878ed226SJulian Elischer 
672878ed226SJulian Elischer static int
673878ed226SJulian Elischer process_info_params(ng_hci_unit_p unit, u_int16_t ocf, struct mbuf *mcp,
674878ed226SJulian Elischer 		struct mbuf *mrp)
675878ed226SJulian Elischer {
676878ed226SJulian Elischer 	int	error = 0, len;
677878ed226SJulian Elischer 
678878ed226SJulian Elischer 	switch (ocf) {
679878ed226SJulian Elischer 	case NG_HCI_OCF_READ_LOCAL_VER:
680878ed226SJulian Elischer 	case NG_HCI_OCF_READ_COUNTRY_CODE:
681878ed226SJulian Elischer 		break;
682878ed226SJulian Elischer 
683878ed226SJulian Elischer 	case NG_HCI_OCF_READ_LOCAL_FEATURES:
684878ed226SJulian Elischer 		m_adj(mrp, sizeof(u_int8_t));
685878ed226SJulian Elischer 		len = min(mrp->m_pkthdr.len, sizeof(unit->features));
686878ed226SJulian Elischer 		m_copydata(mrp, 0, len, (caddr_t) unit->features);
687878ed226SJulian Elischer 		break;
688878ed226SJulian Elischer 
689878ed226SJulian Elischer 	case NG_HCI_OCF_READ_BUFFER_SIZE: {
690878ed226SJulian Elischer 		ng_hci_read_buffer_size_rp	*rp = NULL;
691878ed226SJulian Elischer 
692878ed226SJulian Elischer 		/* Do not update buffer descriptor if node was initialized */
693878ed226SJulian Elischer 		if ((unit->state & NG_HCI_UNIT_READY) == NG_HCI_UNIT_READY)
694878ed226SJulian Elischer 			break;
695878ed226SJulian Elischer 
696878ed226SJulian Elischer 		NG_HCI_M_PULLUP(mrp, sizeof(*rp));
697878ed226SJulian Elischer 		if (mrp != NULL) {
698878ed226SJulian Elischer 			rp = mtod(mrp, ng_hci_read_buffer_size_rp *);
699878ed226SJulian Elischer 
700878ed226SJulian Elischer 			NG_HCI_BUFF_ACL_SET(
701878ed226SJulian Elischer 				unit->buffer,
702878ed226SJulian Elischer 				le16toh(rp->num_acl_pkt),  /* number */
703878ed226SJulian Elischer 				le16toh(rp->max_acl_size), /* size */
704878ed226SJulian Elischer 				le16toh(rp->num_acl_pkt)   /* free */
705878ed226SJulian Elischer 			);
706878ed226SJulian Elischer 
707878ed226SJulian Elischer 			NG_HCI_BUFF_SCO_SET(
708878ed226SJulian Elischer 				unit->buffer,
709878ed226SJulian Elischer 				le16toh(rp->num_sco_pkt), /* number */
710878ed226SJulian Elischer 				rp->max_sco_size,         /* size */
711878ed226SJulian Elischer 				le16toh(rp->num_sco_pkt)  /* free */
712878ed226SJulian Elischer 			);
713878ed226SJulian Elischer 
714878ed226SJulian Elischer 			/* Let upper layers know */
715878ed226SJulian Elischer 			ng_hci_node_is_up(unit->node, unit->acl, NULL, 0);
716878ed226SJulian Elischer 			ng_hci_node_is_up(unit->node, unit->sco, NULL, 0);
717878ed226SJulian Elischer 		} else
718878ed226SJulian Elischer 			error = ENOBUFS;
719878ed226SJulian Elischer 		} break;
720878ed226SJulian Elischer 
721878ed226SJulian Elischer 	case NG_HCI_OCF_READ_BDADDR:
722878ed226SJulian Elischer 		/* Do not update BD_ADDR if node was initialized */
723878ed226SJulian Elischer 		if ((unit->state & NG_HCI_UNIT_READY) == NG_HCI_UNIT_READY)
724878ed226SJulian Elischer 			break;
725878ed226SJulian Elischer 
726878ed226SJulian Elischer 		m_adj(mrp, sizeof(u_int8_t));
727878ed226SJulian Elischer 		len = min(mrp->m_pkthdr.len, sizeof(unit->bdaddr));
728878ed226SJulian Elischer 		m_copydata(mrp, 0, len, (caddr_t) &unit->bdaddr);
729878ed226SJulian Elischer 
730878ed226SJulian Elischer 		/* Let upper layers know */
731878ed226SJulian Elischer 		ng_hci_node_is_up(unit->node, unit->acl, NULL, 0);
732878ed226SJulian Elischer 		ng_hci_node_is_up(unit->node, unit->sco, NULL, 0);
733878ed226SJulian Elischer 		break;
734878ed226SJulian Elischer 
735878ed226SJulian Elischer 	default:
736878ed226SJulian Elischer 		error = EINVAL;
737878ed226SJulian Elischer 		break;
738878ed226SJulian Elischer 	}
739878ed226SJulian Elischer 
740878ed226SJulian Elischer 	NG_FREE_M(mcp);
741878ed226SJulian Elischer 	NG_FREE_M(mrp);
742878ed226SJulian Elischer 
743878ed226SJulian Elischer 	return (error);
744878ed226SJulian Elischer } /* process_info_params */
745878ed226SJulian Elischer 
746878ed226SJulian Elischer /*
747878ed226SJulian Elischer  * Process status command return parameters
748878ed226SJulian Elischer  */
749878ed226SJulian Elischer 
750878ed226SJulian Elischer static int
751878ed226SJulian Elischer process_status_params(ng_hci_unit_p unit, u_int16_t ocf, struct mbuf *mcp,
752878ed226SJulian Elischer 		struct mbuf *mrp)
753878ed226SJulian Elischer {
754878ed226SJulian Elischer 	int	error = 0;
755878ed226SJulian Elischer 
756878ed226SJulian Elischer 	switch (ocf) {
757878ed226SJulian Elischer 	case NG_HCI_OCF_READ_FAILED_CONTACT_CNTR:
758878ed226SJulian Elischer 	case NG_HCI_OCF_RESET_FAILED_CONTACT_CNTR:
759878ed226SJulian Elischer 	case NG_HCI_OCF_GET_LINK_QUALITY:
760878ed226SJulian Elischer 	case NG_HCI_OCF_READ_RSSI:
761878ed226SJulian Elischer 		/* These do not need post processing */
762878ed226SJulian Elischer 		break;
763878ed226SJulian Elischer 
764878ed226SJulian Elischer 	default:
765878ed226SJulian Elischer 		error = EINVAL;
766878ed226SJulian Elischer 		break;
767878ed226SJulian Elischer 	}
768878ed226SJulian Elischer 
769878ed226SJulian Elischer 	NG_FREE_M(mcp);
770878ed226SJulian Elischer 	NG_FREE_M(mrp);
771878ed226SJulian Elischer 
772878ed226SJulian Elischer 	return (error);
773878ed226SJulian Elischer } /* process_status_params */
774878ed226SJulian Elischer 
775878ed226SJulian Elischer /*
776878ed226SJulian Elischer  * Process testing command return parameters
777878ed226SJulian Elischer  */
778878ed226SJulian Elischer 
779878ed226SJulian Elischer int
780878ed226SJulian Elischer process_testing_params(ng_hci_unit_p unit, u_int16_t ocf, struct mbuf *mcp,
781878ed226SJulian Elischer 		struct mbuf *mrp)
782878ed226SJulian Elischer {
783878ed226SJulian Elischer 	int	error = 0;
784878ed226SJulian Elischer 
785878ed226SJulian Elischer 	switch (ocf) {
786878ed226SJulian Elischer 
787878ed226SJulian Elischer 	/*
788878ed226SJulian Elischer 	 * XXX FIXME
789878ed226SJulian Elischer 	 * We do not support these features at this time. However,
790878ed226SJulian Elischer 	 * HCI node could support this and do something smart. At least
791878ed226SJulian Elischer 	 * node can change unit state.
792878ed226SJulian Elischer 	 */
793878ed226SJulian Elischer 
794878ed226SJulian Elischer 	case NG_HCI_OCF_READ_LOOPBACK_MODE:
795878ed226SJulian Elischer 	case NG_HCI_OCF_WRITE_LOOPBACK_MODE:
796878ed226SJulian Elischer 	case NG_HCI_OCF_ENABLE_UNIT_UNDER_TEST:
797878ed226SJulian Elischer 		break;
798878ed226SJulian Elischer 
799878ed226SJulian Elischer 	default:
800878ed226SJulian Elischer 		error = EINVAL;
801878ed226SJulian Elischer 		break;
802878ed226SJulian Elischer 	}
803878ed226SJulian Elischer 
804878ed226SJulian Elischer 	NG_FREE_M(mcp);
805878ed226SJulian Elischer 	NG_FREE_M(mrp);
806878ed226SJulian Elischer 
807878ed226SJulian Elischer 	return (error);
808878ed226SJulian Elischer } /* process_testing_params */
809878ed226SJulian Elischer 
810878ed226SJulian Elischer /*
811*fbc48c2bSTakanori Watanabe  * Process LE command return parameters
812*fbc48c2bSTakanori Watanabe  */
813*fbc48c2bSTakanori Watanabe 
814*fbc48c2bSTakanori Watanabe static int
815*fbc48c2bSTakanori Watanabe process_le_params(ng_hci_unit_p unit, u_int16_t ocf,
816*fbc48c2bSTakanori Watanabe 		struct mbuf *mcp, struct mbuf *mrp)
817*fbc48c2bSTakanori Watanabe {
818*fbc48c2bSTakanori Watanabe 	int	error = 0;
819*fbc48c2bSTakanori Watanabe 
820*fbc48c2bSTakanori Watanabe 	switch (ocf){
821*fbc48c2bSTakanori Watanabe 	case NG_HCI_OCF_LE_SET_EVENT_MASK:
822*fbc48c2bSTakanori Watanabe 	case NG_HCI_OCF_LE_READ_BUFFER_SIZE:
823*fbc48c2bSTakanori Watanabe 	case NG_HCI_OCF_LE_READ_LOCAL_SUPPORTED_FEATURES:
824*fbc48c2bSTakanori Watanabe 	case NG_HCI_OCF_LE_SET_RANDOM_ADDRESS:
825*fbc48c2bSTakanori Watanabe 	case NG_HCI_OCF_LE_SET_ADVERTISING_PARAMETERS:
826*fbc48c2bSTakanori Watanabe 	case NG_HCI_OCF_LE_READ_ADVERTISING_CHANNEL_TX_POWER:
827*fbc48c2bSTakanori Watanabe 	case NG_HCI_OCF_LE_SET_ADVERTISING_DATA:
828*fbc48c2bSTakanori Watanabe 	case NG_HCI_OCF_LE_SET_SCAN_RESPONSE_DATA:
829*fbc48c2bSTakanori Watanabe 	case NG_HCI_OCF_LE_SET_ADVERTISE_ENABLE:
830*fbc48c2bSTakanori Watanabe 	case NG_HCI_OCF_LE_SET_SCAN_PARAMETERS:
831*fbc48c2bSTakanori Watanabe 	case NG_HCI_OCF_LE_SET_SCAN_ENABLE:
832*fbc48c2bSTakanori Watanabe 	case NG_HCI_OCF_LE_CREATE_CONNECTION_CANCEL:
833*fbc48c2bSTakanori Watanabe 	case NG_HCI_OCF_LE_CLEAR_WHITE_LIST:
834*fbc48c2bSTakanori Watanabe 	case NG_HCI_OCF_LE_READ_WHITE_LIST_SIZE:
835*fbc48c2bSTakanori Watanabe 	case NG_HCI_OCF_LE_ADD_DEVICE_TO_WHITE_LIST:
836*fbc48c2bSTakanori Watanabe 	case NG_HCI_OCF_LE_REMOVE_DEVICE_FROM_WHITE_LIST:
837*fbc48c2bSTakanori Watanabe 	case NG_HCI_OCF_LE_SET_HOST_CHANNEL_CLASSIFICATION:
838*fbc48c2bSTakanori Watanabe 	case NG_HCI_OCF_LE_READ_CHANNEL_MAP:
839*fbc48c2bSTakanori Watanabe 	case NG_HCI_OCF_LE_ENCRYPT:
840*fbc48c2bSTakanori Watanabe 	case NG_HCI_OCF_LE_RAND:
841*fbc48c2bSTakanori Watanabe 	case NG_HCI_OCF_LE_LONG_TERM_KEY_REQUEST_REPLY:
842*fbc48c2bSTakanori Watanabe 	case NG_HCI_OCF_LE_LONG_TERM_KEY_REQUEST_NEGATIVE_REPLY:
843*fbc48c2bSTakanori Watanabe 	case NG_HCI_OCF_LE_READ_SUPPORTED_STATUS:
844*fbc48c2bSTakanori Watanabe 	case NG_HCI_OCF_LE_RECEIVER_TEST:
845*fbc48c2bSTakanori Watanabe 	case NG_HCI_OCF_LE_TRANSMITTER_TEST:
846*fbc48c2bSTakanori Watanabe 	case NG_HCI_OCF_LE_TEST_END:
847*fbc48c2bSTakanori Watanabe 
848*fbc48c2bSTakanori Watanabe 		/* These do not need post processing */
849*fbc48c2bSTakanori Watanabe 		break;
850*fbc48c2bSTakanori Watanabe 	case NG_HCI_OCF_LE_CREATE_CONNECTION:
851*fbc48c2bSTakanori Watanabe 	case NG_HCI_OCF_LE_CONNECTION_UPDATE:
852*fbc48c2bSTakanori Watanabe 	case NG_HCI_OCF_LE_READ_REMOTE_USED_FEATURES:
853*fbc48c2bSTakanori Watanabe 	case NG_HCI_OCF_LE_START_ENCRYPTION:
854*fbc48c2bSTakanori Watanabe 
855*fbc48c2bSTakanori Watanabe 
856*fbc48c2bSTakanori Watanabe 	default:
857*fbc48c2bSTakanori Watanabe 		/*
858*fbc48c2bSTakanori Watanabe 		 * None of these command was supposed to generate
859*fbc48c2bSTakanori Watanabe 		 * Command_Complete event. Instead Command_Status event
860*fbc48c2bSTakanori Watanabe 		 * should have been generated and then appropriate event
861*fbc48c2bSTakanori Watanabe 		 * should have been sent to indicate the final result.
862*fbc48c2bSTakanori Watanabe 		 */
863*fbc48c2bSTakanori Watanabe 
864*fbc48c2bSTakanori Watanabe 		error = EINVAL;
865*fbc48c2bSTakanori Watanabe 		break;
866*fbc48c2bSTakanori Watanabe 	}
867*fbc48c2bSTakanori Watanabe 
868*fbc48c2bSTakanori Watanabe 	NG_FREE_M(mcp);
869*fbc48c2bSTakanori Watanabe 	NG_FREE_M(mrp);
870*fbc48c2bSTakanori Watanabe 
871*fbc48c2bSTakanori Watanabe 	return (error);
872*fbc48c2bSTakanori Watanabe 
873*fbc48c2bSTakanori Watanabe }
874*fbc48c2bSTakanori Watanabe 
875*fbc48c2bSTakanori Watanabe 
876*fbc48c2bSTakanori Watanabe 
877*fbc48c2bSTakanori Watanabe static int
878*fbc48c2bSTakanori Watanabe process_le_status(ng_hci_unit_p unit,ng_hci_command_status_ep *ep,
879*fbc48c2bSTakanori Watanabe 		struct mbuf *mcp)
880*fbc48c2bSTakanori Watanabe {
881*fbc48c2bSTakanori Watanabe 	int	error = 0;
882*fbc48c2bSTakanori Watanabe 
883*fbc48c2bSTakanori Watanabe 	switch (NG_HCI_OCF(ep->opcode)){
884*fbc48c2bSTakanori Watanabe 	case NG_HCI_OCF_LE_CREATE_CONNECTION:
885*fbc48c2bSTakanori Watanabe 	case NG_HCI_OCF_LE_CONNECTION_UPDATE:
886*fbc48c2bSTakanori Watanabe 	case NG_HCI_OCF_LE_READ_REMOTE_USED_FEATURES:
887*fbc48c2bSTakanori Watanabe 	case NG_HCI_OCF_LE_START_ENCRYPTION:
888*fbc48c2bSTakanori Watanabe 
889*fbc48c2bSTakanori Watanabe 		/* These do not need post processing */
890*fbc48c2bSTakanori Watanabe 		break;
891*fbc48c2bSTakanori Watanabe 
892*fbc48c2bSTakanori Watanabe 	case NG_HCI_OCF_LE_SET_EVENT_MASK:
893*fbc48c2bSTakanori Watanabe 	case NG_HCI_OCF_LE_READ_BUFFER_SIZE:
894*fbc48c2bSTakanori Watanabe 	case NG_HCI_OCF_LE_READ_LOCAL_SUPPORTED_FEATURES:
895*fbc48c2bSTakanori Watanabe 	case NG_HCI_OCF_LE_SET_RANDOM_ADDRESS:
896*fbc48c2bSTakanori Watanabe 	case NG_HCI_OCF_LE_SET_ADVERTISING_PARAMETERS:
897*fbc48c2bSTakanori Watanabe 	case NG_HCI_OCF_LE_READ_ADVERTISING_CHANNEL_TX_POWER:
898*fbc48c2bSTakanori Watanabe 	case NG_HCI_OCF_LE_SET_ADVERTISING_DATA:
899*fbc48c2bSTakanori Watanabe 	case NG_HCI_OCF_LE_SET_SCAN_RESPONSE_DATA:
900*fbc48c2bSTakanori Watanabe 	case NG_HCI_OCF_LE_SET_ADVERTISE_ENABLE:
901*fbc48c2bSTakanori Watanabe 	case NG_HCI_OCF_LE_SET_SCAN_PARAMETERS:
902*fbc48c2bSTakanori Watanabe 	case NG_HCI_OCF_LE_SET_SCAN_ENABLE:
903*fbc48c2bSTakanori Watanabe 	case NG_HCI_OCF_LE_CREATE_CONNECTION_CANCEL:
904*fbc48c2bSTakanori Watanabe 	case NG_HCI_OCF_LE_CLEAR_WHITE_LIST:
905*fbc48c2bSTakanori Watanabe 	case NG_HCI_OCF_LE_READ_WHITE_LIST_SIZE:
906*fbc48c2bSTakanori Watanabe 	case NG_HCI_OCF_LE_ADD_DEVICE_TO_WHITE_LIST:
907*fbc48c2bSTakanori Watanabe 	case NG_HCI_OCF_LE_REMOVE_DEVICE_FROM_WHITE_LIST:
908*fbc48c2bSTakanori Watanabe 	case NG_HCI_OCF_LE_SET_HOST_CHANNEL_CLASSIFICATION:
909*fbc48c2bSTakanori Watanabe 	case NG_HCI_OCF_LE_READ_CHANNEL_MAP:
910*fbc48c2bSTakanori Watanabe 	case NG_HCI_OCF_LE_ENCRYPT:
911*fbc48c2bSTakanori Watanabe 	case NG_HCI_OCF_LE_RAND:
912*fbc48c2bSTakanori Watanabe 	case NG_HCI_OCF_LE_LONG_TERM_KEY_REQUEST_REPLY:
913*fbc48c2bSTakanori Watanabe 	case NG_HCI_OCF_LE_LONG_TERM_KEY_REQUEST_NEGATIVE_REPLY:
914*fbc48c2bSTakanori Watanabe 	case NG_HCI_OCF_LE_READ_SUPPORTED_STATUS:
915*fbc48c2bSTakanori Watanabe 	case NG_HCI_OCF_LE_RECEIVER_TEST:
916*fbc48c2bSTakanori Watanabe 	case NG_HCI_OCF_LE_TRANSMITTER_TEST:
917*fbc48c2bSTakanori Watanabe 	case NG_HCI_OCF_LE_TEST_END:
918*fbc48c2bSTakanori Watanabe 
919*fbc48c2bSTakanori Watanabe 
920*fbc48c2bSTakanori Watanabe 	default:
921*fbc48c2bSTakanori Watanabe 		/*
922*fbc48c2bSTakanori Watanabe 		 * None of these command was supposed to generate
923*fbc48c2bSTakanori Watanabe 		 * Command_Stutus event. Command Complete instead.
924*fbc48c2bSTakanori Watanabe 		 */
925*fbc48c2bSTakanori Watanabe 
926*fbc48c2bSTakanori Watanabe 		error = EINVAL;
927*fbc48c2bSTakanori Watanabe 		break;
928*fbc48c2bSTakanori Watanabe 	}
929*fbc48c2bSTakanori Watanabe 
930*fbc48c2bSTakanori Watanabe 	NG_FREE_M(mcp);
931*fbc48c2bSTakanori Watanabe 
932*fbc48c2bSTakanori Watanabe 	return (error);
933*fbc48c2bSTakanori Watanabe 
934*fbc48c2bSTakanori Watanabe }
935*fbc48c2bSTakanori Watanabe 
936*fbc48c2bSTakanori Watanabe /*
937878ed226SJulian Elischer  * Process link control command status
938878ed226SJulian Elischer  */
939878ed226SJulian Elischer 
940878ed226SJulian Elischer static int
941878ed226SJulian Elischer process_link_control_status(ng_hci_unit_p unit, ng_hci_command_status_ep *ep,
942878ed226SJulian Elischer 		struct mbuf *mcp)
943878ed226SJulian Elischer {
944878ed226SJulian Elischer 	int	error = 0;
945878ed226SJulian Elischer 
946878ed226SJulian Elischer 	switch (NG_HCI_OCF(ep->opcode)) {
947878ed226SJulian Elischer 	case NG_HCI_OCF_INQUIRY:
948878ed226SJulian Elischer 	case NG_HCI_OCF_DISCON:		/* XXX */
949878ed226SJulian Elischer 	case NG_HCI_OCF_REJECT_CON:	/* XXX */
950878ed226SJulian Elischer 	case NG_HCI_OCF_CHANGE_CON_PKT_TYPE:
951878ed226SJulian Elischer 	case NG_HCI_OCF_AUTH_REQ:
952878ed226SJulian Elischer 	case NG_HCI_OCF_SET_CON_ENCRYPTION:
953878ed226SJulian Elischer 	case NG_HCI_OCF_CHANGE_CON_LINK_KEY:
954878ed226SJulian Elischer 	case NG_HCI_OCF_MASTER_LINK_KEY:
955878ed226SJulian Elischer 	case NG_HCI_OCF_REMOTE_NAME_REQ:
956878ed226SJulian Elischer 	case NG_HCI_OCF_READ_REMOTE_FEATURES:
957878ed226SJulian Elischer 	case NG_HCI_OCF_READ_REMOTE_VER_INFO:
958878ed226SJulian Elischer 	case NG_HCI_OCF_READ_CLOCK_OFFSET:
959878ed226SJulian Elischer 		/* These do not need post processing */
960878ed226SJulian Elischer 		break;
961878ed226SJulian Elischer 
962878ed226SJulian Elischer 	case NG_HCI_OCF_CREATE_CON:
963878ed226SJulian Elischer 		break;
964878ed226SJulian Elischer 
965878ed226SJulian Elischer 	case NG_HCI_OCF_ADD_SCO_CON:
966878ed226SJulian Elischer 		break;
967878ed226SJulian Elischer 
968878ed226SJulian Elischer 	case NG_HCI_OCF_ACCEPT_CON:
969878ed226SJulian Elischer 		break;
970878ed226SJulian Elischer 
971878ed226SJulian Elischer 	case NG_HCI_OCF_INQUIRY_CANCEL:
972878ed226SJulian Elischer 	case NG_HCI_OCF_PERIODIC_INQUIRY:
973878ed226SJulian Elischer 	case NG_HCI_OCF_EXIT_PERIODIC_INQUIRY:
974878ed226SJulian Elischer 	case NG_HCI_OCF_LINK_KEY_REP:
975878ed226SJulian Elischer 	case NG_HCI_OCF_LINK_KEY_NEG_REP:
976878ed226SJulian Elischer 	case NG_HCI_OCF_PIN_CODE_REP:
977878ed226SJulian Elischer 	case NG_HCI_OCF_PIN_CODE_NEG_REP:
978878ed226SJulian Elischer 	default:
979878ed226SJulian Elischer 
980878ed226SJulian Elischer 		/*
981878ed226SJulian Elischer 		 * None of these command was supposed to generate
982878ed226SJulian Elischer 		 * Command_Status event. Instead Command_Complete event
983878ed226SJulian Elischer 		 * should have been sent.
984878ed226SJulian Elischer 		 */
985878ed226SJulian Elischer 
986878ed226SJulian Elischer 		error = EINVAL;
987878ed226SJulian Elischer 		break;
988878ed226SJulian Elischer 	}
989878ed226SJulian Elischer 
990878ed226SJulian Elischer 	NG_FREE_M(mcp);
991878ed226SJulian Elischer 
992878ed226SJulian Elischer 	return (error);
993878ed226SJulian Elischer } /* process_link_control_status */
994878ed226SJulian Elischer 
995878ed226SJulian Elischer /*
996878ed226SJulian Elischer  * Process link policy command status
997878ed226SJulian Elischer  */
998878ed226SJulian Elischer 
999878ed226SJulian Elischer static int
1000878ed226SJulian Elischer process_link_policy_status(ng_hci_unit_p unit, ng_hci_command_status_ep *ep,
1001878ed226SJulian Elischer 		struct mbuf *mcp)
1002878ed226SJulian Elischer {
1003878ed226SJulian Elischer 	int	error = 0;
1004878ed226SJulian Elischer 
1005878ed226SJulian Elischer 	switch (NG_HCI_OCF(ep->opcode)) {
1006878ed226SJulian Elischer 	case NG_HCI_OCF_HOLD_MODE:
1007878ed226SJulian Elischer 	case NG_HCI_OCF_SNIFF_MODE:
1008878ed226SJulian Elischer 	case NG_HCI_OCF_EXIT_SNIFF_MODE:
1009878ed226SJulian Elischer 	case NG_HCI_OCF_PARK_MODE:
1010878ed226SJulian Elischer 	case NG_HCI_OCF_EXIT_PARK_MODE:
1011878ed226SJulian Elischer 	case NG_HCI_OCF_SWITCH_ROLE:
1012878ed226SJulian Elischer 		/* These do not need post processing */
1013878ed226SJulian Elischer 		break;
1014878ed226SJulian Elischer 
1015878ed226SJulian Elischer 	case NG_HCI_OCF_QOS_SETUP:
1016878ed226SJulian Elischer 		break;
1017878ed226SJulian Elischer 
1018878ed226SJulian Elischer 	case NG_HCI_OCF_ROLE_DISCOVERY:
1019878ed226SJulian Elischer 	case NG_HCI_OCF_READ_LINK_POLICY_SETTINGS:
1020878ed226SJulian Elischer 	case NG_HCI_OCF_WRITE_LINK_POLICY_SETTINGS:
1021878ed226SJulian Elischer 	default:
1022878ed226SJulian Elischer 
1023878ed226SJulian Elischer 		/*
1024878ed226SJulian Elischer 		 * None of these command was supposed to generate
1025878ed226SJulian Elischer 		 * Command_Status event. Instead Command_Complete event
1026878ed226SJulian Elischer 		 * should have been sent.
1027878ed226SJulian Elischer 		 */
1028878ed226SJulian Elischer 
1029878ed226SJulian Elischer 		error = EINVAL;
1030878ed226SJulian Elischer 		break;
1031878ed226SJulian Elischer 	}
1032878ed226SJulian Elischer 
1033878ed226SJulian Elischer 	NG_FREE_M(mcp);
1034878ed226SJulian Elischer 
1035878ed226SJulian Elischer 	return (error);
1036878ed226SJulian Elischer } /* process_link_policy_status */
1037878ed226SJulian Elischer 
1038