1 /*
2 * ng_hci_cmds.c
3 */
4
5 /*-
6 * SPDX-License-Identifier: BSD-2-Clause-FreeBSD
7 *
8 * Copyright (c) Maksim Yevmenkin <m_evmenkin@yahoo.com>
9 * All rights reserved.
10 *
11 * Redistribution and use in source and binary forms, with or without
12 * modification, are permitted provided that the following conditions
13 * are met:
14 * 1. Redistributions of source code must retain the above copyright
15 * notice, this list of conditions and the following disclaimer.
16 * 2. Redistributions in binary form must reproduce the above copyright
17 * notice, this list of conditions and the following disclaimer in the
18 * documentation and/or other materials provided with the distribution.
19 *
20 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
21 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
22 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
23 * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
24 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
25 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
26 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
27 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
28 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
29 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
30 * SUCH DAMAGE.
31 *
32 * $Id: ng_hci_cmds.c,v 1.4 2003/09/08 18:57:51 max Exp $
33 * $FreeBSD$
34 */
35
36 #include <sys/param.h>
37 #include <sys/systm.h>
38 #include <sys/kernel.h>
39 #include <sys/endian.h>
40 #include <sys/malloc.h>
41 #include <sys/mbuf.h>
42 #include <sys/queue.h>
43 #include <netgraph/ng_message.h>
44 #include <netgraph/netgraph.h>
45 #include <netgraph/bluetooth/include/ng_bluetooth.h>
46 #include <netgraph/bluetooth/include/ng_hci.h>
47 #include <netgraph/bluetooth/hci/ng_hci_var.h>
48 #include <netgraph/bluetooth/hci/ng_hci_cmds.h>
49 #include <netgraph/bluetooth/hci/ng_hci_evnt.h>
50 #include <netgraph/bluetooth/hci/ng_hci_ulpi.h>
51 #include <netgraph/bluetooth/hci/ng_hci_misc.h>
52
53 /******************************************************************************
54 ******************************************************************************
55 ** HCI commands processing module
56 ******************************************************************************
57 ******************************************************************************/
58
59 #undef min
60 #define min(a, b) ((a) < (b))? (a) : (b)
61
62 static int complete_command (ng_hci_unit_p, int, struct mbuf **);
63
64 static int process_link_control_params
65 (ng_hci_unit_p, u_int16_t, struct mbuf *, struct mbuf *);
66 static int process_link_policy_params
67 (ng_hci_unit_p, u_int16_t, struct mbuf *, struct mbuf *);
68 static int process_hc_baseband_params
69 (ng_hci_unit_p, u_int16_t, struct mbuf *, struct mbuf *);
70 static int process_info_params
71 (ng_hci_unit_p, u_int16_t, struct mbuf *, struct mbuf *);
72 static int process_status_params
73 (ng_hci_unit_p, u_int16_t, struct mbuf *, struct mbuf *);
74 static int process_testing_params
75 (ng_hci_unit_p, u_int16_t, struct mbuf *, struct mbuf *);
76 static int process_le_params
77 (ng_hci_unit_p, u_int16_t, struct mbuf *, struct mbuf *);
78
79 static int process_link_control_status
80 (ng_hci_unit_p, ng_hci_command_status_ep *, struct mbuf *);
81 static int process_link_policy_status
82 (ng_hci_unit_p, ng_hci_command_status_ep *, struct mbuf *);
83 static int process_le_status
84 (ng_hci_unit_p, ng_hci_command_status_ep *, struct mbuf *);
85
86 /*
87 * Send HCI command to the driver.
88 */
89
90 int
91 ng_hci_send_command(ng_hci_unit_p unit)
92 {
93 struct mbuf *m0 = NULL, *m = NULL;
94 int free, error = 0;
95
96 /* Check if other command is pending */
97 if (unit->state & NG_HCI_UNIT_COMMAND_PENDING)
98 return (0);
99
100 /* Check if unit can accept our command */
101 NG_HCI_BUFF_CMD_GET(unit->buffer, free);
102 if (free == 0)
103 return (0);
104
105 /* Check if driver hook is still ok */
106 if (unit->drv == NULL || NG_HOOK_NOT_VALID(unit->drv)) {
107 NG_HCI_WARN(
108 "%s: %s - hook \"%s\" is not connected or valid\n",
109 __func__, NG_NODE_NAME(unit->node), NG_HCI_HOOK_DRV);
110
111 NG_BT_MBUFQ_DRAIN(&unit->cmdq);
112
113 return (ENOTCONN);
114 }
115
116 /*
117 * Get first command from queue, give it to RAW hook then
118 * make copy of it and send it to the driver
119 */
120
121 m0 = NG_BT_MBUFQ_FIRST(&unit->cmdq);
122 if (m0 == NULL)
123 return (0);
124
125 ng_hci_mtap(unit, m0);
126
127 m = m_dup(m0, M_NOWAIT);
128 if (m != NULL)
129 NG_SEND_DATA_ONLY(error, unit->drv, m);
130 else
131 error = ENOBUFS;
132
133 if (error != 0)
134 NG_HCI_ERR(
135 "%s: %s - could not send HCI command, error=%d\n",
136 __func__, NG_NODE_NAME(unit->node), error);
137
138 /*
139 * Even if we were not able to send command we still pretend
140 * that everything is OK and let timeout handle that.
141 */
142
143 NG_HCI_BUFF_CMD_USE(unit->buffer, 1);
144 NG_HCI_STAT_CMD_SENT(unit->stat);
145 NG_HCI_STAT_BYTES_SENT(unit->stat, m0->m_pkthdr.len);
146
147 /*
148 * Note: ng_hci_command_timeout() will set
149 * NG_HCI_UNIT_COMMAND_PENDING flag
150 */
151
152 ng_hci_command_timeout(unit);
153
154 return (0);
155 } /* ng_hci_send_command */
156
157 /*
158 * Process HCI Command_Compete event. Complete HCI command, and do post
159 * processing on the command parameters (cp) and command return parameters
160 * (e) if required (for example adjust state).
161 */
162
163 int
164 ng_hci_process_command_complete(ng_hci_unit_p unit, struct mbuf *e)
165 {
166 ng_hci_command_compl_ep *ep = NULL;
167 struct mbuf *cp = NULL;
168 int error = 0;
169
170 /* Get event packet and update command buffer info */
171 NG_HCI_M_PULLUP(e, sizeof(*ep));
172 if (e == NULL)
173 return (ENOBUFS); /* XXX this is bad */
174
175 ep = mtod(e, ng_hci_command_compl_ep *);
176 NG_HCI_BUFF_CMD_SET(unit->buffer, ep->num_cmd_pkts);
177
178 /* Check for special NOOP command */
179 if (ep->opcode == 0x0000) {
180 NG_FREE_M(e);
181 goto out;
182 }
183
184 /* Try to match first command item in the queue */
185 error = complete_command(unit, ep->opcode, &cp);
186 if (error != 0) {
187 NG_FREE_M(e);
188 goto out;
189 }
190
191 /*
192 * Perform post processing on command parameters and return parameters
193 * do it only if status is OK (status == 0). Status is the first byte
194 * of any command return parameters.
195 */
196
197 ep->opcode = le16toh(ep->opcode);
198 m_adj(e, sizeof(*ep));
199
200 if (*mtod(e, u_int8_t *) == 0) { /* XXX m_pullup here? */
201 switch (NG_HCI_OGF(ep->opcode)) {
202 case NG_HCI_OGF_LINK_CONTROL:
203 error = process_link_control_params(unit,
204 NG_HCI_OCF(ep->opcode), cp, e);
205 break;
206
207 case NG_HCI_OGF_LINK_POLICY:
208 error = process_link_policy_params(unit,
209 NG_HCI_OCF(ep->opcode), cp, e);
210 break;
211
212 case NG_HCI_OGF_HC_BASEBAND:
213 error = process_hc_baseband_params(unit,
214 NG_HCI_OCF(ep->opcode), cp, e);
215 break;
216
217 case NG_HCI_OGF_INFO:
218 error = process_info_params(unit,
219 NG_HCI_OCF(ep->opcode), cp, e);
220 break;
221
222 case NG_HCI_OGF_STATUS:
223 error = process_status_params(unit,
224 NG_HCI_OCF(ep->opcode), cp, e);
225 break;
226
227 case NG_HCI_OGF_TESTING:
228 error = process_testing_params(unit,
229 NG_HCI_OCF(ep->opcode), cp, e);
230 break;
231 case NG_HCI_OGF_LE:
232 error = process_le_params(unit,
233 NG_HCI_OCF(ep->opcode), cp, e);
234 break;
235 case NG_HCI_OGF_BT_LOGO:
236 case NG_HCI_OGF_VENDOR:
237 NG_FREE_M(cp);
238 NG_FREE_M(e);
239 break;
240
241 default:
242 NG_FREE_M(cp);
243 NG_FREE_M(e);
244 error = EINVAL;
245 break;
246 }
247 } else {
248 NG_HCI_ERR(
249 "%s: %s - HCI command failed, OGF=%#x, OCF=%#x, status=%#x\n",
250 __func__, NG_NODE_NAME(unit->node),
251 NG_HCI_OGF(ep->opcode), NG_HCI_OCF(ep->opcode),
252 *mtod(e, u_int8_t *));
253
254 NG_FREE_M(cp);
255 NG_FREE_M(e);
256 }
257 out:
258 ng_hci_send_command(unit);
259
260 return (error);
261 } /* ng_hci_process_command_complete */
262
263 /*
264 * Process HCI Command_Status event. Check the status (mst) and do post
265 * processing (if required).
266 */
267
268 int
269 ng_hci_process_command_status(ng_hci_unit_p unit, struct mbuf *e)
270 {
271 ng_hci_command_status_ep *ep = NULL;
272 struct mbuf *cp = NULL;
273 int error = 0;
274
275 /* Update command buffer info */
276 NG_HCI_M_PULLUP(e, sizeof(*ep));
277 if (e == NULL)
278 return (ENOBUFS); /* XXX this is bad */
279
280 ep = mtod(e, ng_hci_command_status_ep *);
281 NG_HCI_BUFF_CMD_SET(unit->buffer, ep->num_cmd_pkts);
282
283 /* Check for special NOOP command */
284 if (ep->opcode == 0x0000)
285 goto out;
286
287 /* Try to match first command item in the queue */
288 error = complete_command(unit, ep->opcode, &cp);
289 if (error != 0)
290 goto out;
291
292 /*
293 * Perform post processing on HCI Command_Status event
294 */
295
296 ep->opcode = le16toh(ep->opcode);
297
298 switch (NG_HCI_OGF(ep->opcode)) {
299 case NG_HCI_OGF_LINK_CONTROL:
300 error = process_link_control_status(unit, ep, cp);
301 break;
302
303 case NG_HCI_OGF_LINK_POLICY:
304 error = process_link_policy_status(unit, ep, cp);
305 break;
306 case NG_HCI_OGF_LE:
307 error = process_le_status(unit, ep, cp);
308 break;
309 case NG_HCI_OGF_BT_LOGO:
310 case NG_HCI_OGF_VENDOR:
311 NG_FREE_M(cp);
312 break;
313
314 case NG_HCI_OGF_HC_BASEBAND:
315 case NG_HCI_OGF_INFO:
316 case NG_HCI_OGF_STATUS:
317 case NG_HCI_OGF_TESTING:
318 default:
319 NG_FREE_M(cp);
320 error = EINVAL;
321 break;
322 }
323 out:
324 NG_FREE_M(e);
325 ng_hci_send_command(unit);
326
327 return (error);
328 } /* ng_hci_process_command_status */
329
330 /*
331 * Complete queued HCI command.
332 */
333
334 static int
335 complete_command(ng_hci_unit_p unit, int opcode, struct mbuf **cp)
336 {
337 struct mbuf *m = NULL;
338
339 /* Check unit state */
340 if (!(unit->state & NG_HCI_UNIT_COMMAND_PENDING)) {
341 NG_HCI_ALERT(
342 "%s: %s - no pending command, state=%#x\n",
343 __func__, NG_NODE_NAME(unit->node), unit->state);
344
345 return (EINVAL);
346 }
347
348 /* Get first command in the queue */
349 m = NG_BT_MBUFQ_FIRST(&unit->cmdq);
350 if (m == NULL) {
351 NG_HCI_ALERT(
352 "%s: %s - empty command queue?!\n", __func__, NG_NODE_NAME(unit->node));
353
354 return (EINVAL);
355 }
356
357 /*
358 * Match command opcode, if does not match - do nothing and
359 * let timeout handle that.
360 */
361
362 if (mtod(m, ng_hci_cmd_pkt_t *)->opcode != opcode) {
363 NG_HCI_ALERT(
364 "%s: %s - command queue is out of sync\n", __func__, NG_NODE_NAME(unit->node));
365
366 return (EINVAL);
367 }
368
369 /*
370 * Now we can remove command timeout, dequeue completed command
371 * and return command parameters. ng_hci_command_untimeout will
372 * drop NG_HCI_UNIT_COMMAND_PENDING flag.
373 * Note: if ng_hci_command_untimeout() fails (returns non-zero)
374 * then timeout already happened and timeout message went info node
375 * queue. In this case we ignore command completion and pretend
376 * there is a timeout.
377 */
378
379 if (ng_hci_command_untimeout(unit) != 0)
380 return (ETIMEDOUT);
381
382 NG_BT_MBUFQ_DEQUEUE(&unit->cmdq, *cp);
383 m_adj(*cp, sizeof(ng_hci_cmd_pkt_t));
384
385 return (0);
386 } /* complete_command */
387
388 /*
389 * Process HCI command timeout
390 */
391
392 void
393 ng_hci_process_command_timeout(node_p node, hook_p hook, void *arg1, int arg2)
394 {
395 ng_hci_unit_p unit = NULL;
396 struct mbuf *m = NULL;
397 u_int16_t opcode;
398
399 if (NG_NODE_NOT_VALID(node)) {
400 printf("%s: Netgraph node is not valid\n", __func__);
401 return;
402 }
403
404 unit = (ng_hci_unit_p) NG_NODE_PRIVATE(node);
405
406 if (unit->state & NG_HCI_UNIT_COMMAND_PENDING) {
407 unit->state &= ~NG_HCI_UNIT_COMMAND_PENDING;
408
409 NG_BT_MBUFQ_DEQUEUE(&unit->cmdq, m);
410 if (m == NULL) {
411 NG_HCI_ALERT(
412 "%s: %s - command queue is out of sync!\n", __func__, NG_NODE_NAME(unit->node));
413
414 return;
415 }
416
417 opcode = le16toh(mtod(m, ng_hci_cmd_pkt_t *)->opcode);
418 NG_FREE_M(m);
419
420 NG_HCI_ERR(
421 "%s: %s - unable to complete HCI command OGF=%#x, OCF=%#x. Timeout\n",
422 __func__, NG_NODE_NAME(unit->node), NG_HCI_OGF(opcode),
423 NG_HCI_OCF(opcode));
424
425 /* Try to send more commands */
426 NG_HCI_BUFF_CMD_SET(unit->buffer, 1);
427 ng_hci_send_command(unit);
428 } else
429 NG_HCI_ALERT(
430 "%s: %s - no pending command\n", __func__, NG_NODE_NAME(unit->node));
431 } /* ng_hci_process_command_timeout */
432
433 /*
434 * Process link command return parameters
435 */
436
437 static int
438 process_link_control_params(ng_hci_unit_p unit, u_int16_t ocf,
439 struct mbuf *mcp, struct mbuf *mrp)
440 {
441 int error = 0;
442
443 switch (ocf) {
444 case NG_HCI_OCF_INQUIRY_CANCEL:
445 case NG_HCI_OCF_PERIODIC_INQUIRY:
446 case NG_HCI_OCF_EXIT_PERIODIC_INQUIRY:
447 case NG_HCI_OCF_LINK_KEY_REP:
448 case NG_HCI_OCF_LINK_KEY_NEG_REP:
449 case NG_HCI_OCF_PIN_CODE_REP:
450 case NG_HCI_OCF_PIN_CODE_NEG_REP:
451 /* These do not need post processing */
452 break;
453
454 case NG_HCI_OCF_INQUIRY:
455 case NG_HCI_OCF_CREATE_CON:
456 case NG_HCI_OCF_DISCON:
457 case NG_HCI_OCF_ADD_SCO_CON:
458 case NG_HCI_OCF_ACCEPT_CON:
459 case NG_HCI_OCF_REJECT_CON:
460 case NG_HCI_OCF_CHANGE_CON_PKT_TYPE:
461 case NG_HCI_OCF_AUTH_REQ:
462 case NG_HCI_OCF_SET_CON_ENCRYPTION:
463 case NG_HCI_OCF_CHANGE_CON_LINK_KEY:
464 case NG_HCI_OCF_MASTER_LINK_KEY:
465 case NG_HCI_OCF_REMOTE_NAME_REQ:
466 case NG_HCI_OCF_READ_REMOTE_FEATURES:
467 case NG_HCI_OCF_READ_REMOTE_VER_INFO:
468 case NG_HCI_OCF_READ_CLOCK_OFFSET:
469 default:
470
471 /*
472 * None of these command was supposed to generate
473 * Command_Complete event. Instead Command_Status event
474 * should have been generated and then appropriate event
475 * should have been sent to indicate the final result.
476 */
477
478 error = EINVAL;
479 break;
480 }
481
482 NG_FREE_M(mcp);
483 NG_FREE_M(mrp);
484
485 return (error);
486 } /* process_link_control_params */
487
488 /*
489 * Process link policy command return parameters
490 */
491
492 static int
493 process_link_policy_params(ng_hci_unit_p unit, u_int16_t ocf,
494 struct mbuf *mcp, struct mbuf *mrp)
495 {
496 int error = 0;
497
498 switch (ocf){
499 case NG_HCI_OCF_ROLE_DISCOVERY: {
500 ng_hci_role_discovery_rp *rp = NULL;
501 ng_hci_unit_con_t *con = NULL;
502 u_int16_t h;
503
504 NG_HCI_M_PULLUP(mrp, sizeof(*rp));
505 if (mrp != NULL) {
506 rp = mtod(mrp, ng_hci_role_discovery_rp *);
507
508 h = NG_HCI_CON_HANDLE(le16toh(rp->con_handle));
509 con = ng_hci_con_by_handle(unit, h);
510 if (con == NULL) {
511 NG_HCI_ALERT(
512 "%s: %s - invalid connection handle=%d\n",
513 __func__, NG_NODE_NAME(unit->node), h);
514 error = ENOENT;
515 } else if (con->link_type != NG_HCI_LINK_ACL) {
516 NG_HCI_ALERT(
517 "%s: %s - invalid link type=%d\n", __func__, NG_NODE_NAME(unit->node),
518 con->link_type);
519 error = EINVAL;
520 } else
521 con->role = rp->role;
522 } else
523 error = ENOBUFS;
524 } break;
525
526 case NG_HCI_OCF_READ_LINK_POLICY_SETTINGS:
527 case NG_HCI_OCF_WRITE_LINK_POLICY_SETTINGS:
528 /* These do not need post processing */
529 break;
530
531 case NG_HCI_OCF_HOLD_MODE:
532 case NG_HCI_OCF_SNIFF_MODE:
533 case NG_HCI_OCF_EXIT_SNIFF_MODE:
534 case NG_HCI_OCF_PARK_MODE:
535 case NG_HCI_OCF_EXIT_PARK_MODE:
536 case NG_HCI_OCF_QOS_SETUP:
537 case NG_HCI_OCF_SWITCH_ROLE:
538 default:
539
540 /*
541 * None of these command was supposed to generate
542 * Command_Complete event. Instead Command_Status event
543 * should have been generated and then appropriate event
544 * should have been sent to indicate the final result.
545 */
546
547 error = EINVAL;
548 break;
549 }
550
551 NG_FREE_M(mcp);
552 NG_FREE_M(mrp);
553
554 return (error);
555 } /* process_link_policy_params */
556
557 /*
558 * Process HC and baseband command return parameters
559 */
560
561 int
562 process_hc_baseband_params(ng_hci_unit_p unit, u_int16_t ocf,
563 struct mbuf *mcp, struct mbuf *mrp)
564 {
565 int error = 0;
566
567 switch (ocf) {
568 case NG_HCI_OCF_SET_EVENT_MASK:
569 case NG_HCI_OCF_SET_EVENT_FILTER:
570 case NG_HCI_OCF_FLUSH: /* XXX Do we need to handle that? */
571 case NG_HCI_OCF_READ_PIN_TYPE:
572 case NG_HCI_OCF_WRITE_PIN_TYPE:
573 case NG_HCI_OCF_CREATE_NEW_UNIT_KEY:
574 case NG_HCI_OCF_WRITE_STORED_LINK_KEY:
575 case NG_HCI_OCF_WRITE_CON_ACCEPT_TIMO:
576 case NG_HCI_OCF_WRITE_PAGE_TIMO:
577 case NG_HCI_OCF_READ_SCAN_ENABLE:
578 case NG_HCI_OCF_WRITE_SCAN_ENABLE:
579 case NG_HCI_OCF_WRITE_PAGE_SCAN_ACTIVITY:
580 case NG_HCI_OCF_WRITE_INQUIRY_SCAN_ACTIVITY:
581 case NG_HCI_OCF_READ_AUTH_ENABLE:
582 case NG_HCI_OCF_WRITE_AUTH_ENABLE:
583 case NG_HCI_OCF_READ_ENCRYPTION_MODE:
584 case NG_HCI_OCF_WRITE_ENCRYPTION_MODE:
585 case NG_HCI_OCF_WRITE_VOICE_SETTINGS:
586 case NG_HCI_OCF_READ_NUM_BROADCAST_RETRANS:
587 case NG_HCI_OCF_WRITE_NUM_BROADCAST_RETRANS:
588 case NG_HCI_OCF_READ_HOLD_MODE_ACTIVITY:
589 case NG_HCI_OCF_WRITE_HOLD_MODE_ACTIVITY:
590 case NG_HCI_OCF_READ_SCO_FLOW_CONTROL:
591 case NG_HCI_OCF_WRITE_SCO_FLOW_CONTROL:
592 case NG_HCI_OCF_H2HC_FLOW_CONTROL: /* XXX Not supported this time */
593 case NG_HCI_OCF_HOST_BUFFER_SIZE:
594 case NG_HCI_OCF_READ_IAC_LAP:
595 case NG_HCI_OCF_WRITE_IAC_LAP:
596 case NG_HCI_OCF_READ_PAGE_SCAN_PERIOD:
597 case NG_HCI_OCF_WRITE_PAGE_SCAN_PERIOD:
598 case NG_HCI_OCF_READ_PAGE_SCAN:
599 case NG_HCI_OCF_WRITE_PAGE_SCAN:
600 case NG_HCI_OCF_READ_LINK_SUPERVISION_TIMO:
601 case NG_HCI_OCF_WRITE_LINK_SUPERVISION_TIMO:
602 case NG_HCI_OCF_READ_SUPPORTED_IAC_NUM:
603 case NG_HCI_OCF_READ_STORED_LINK_KEY:
604 case NG_HCI_OCF_DELETE_STORED_LINK_KEY:
605 case NG_HCI_OCF_READ_CON_ACCEPT_TIMO:
606 case NG_HCI_OCF_READ_PAGE_TIMO:
607 case NG_HCI_OCF_READ_PAGE_SCAN_ACTIVITY:
608 case NG_HCI_OCF_READ_INQUIRY_SCAN_ACTIVITY:
609 case NG_HCI_OCF_READ_VOICE_SETTINGS:
610 case NG_HCI_OCF_READ_AUTO_FLUSH_TIMO:
611 case NG_HCI_OCF_WRITE_AUTO_FLUSH_TIMO:
612 case NG_HCI_OCF_READ_XMIT_LEVEL:
613 case NG_HCI_OCF_HOST_NUM_COMPL_PKTS: /* XXX Can get here? */
614 case NG_HCI_OCF_CHANGE_LOCAL_NAME:
615 case NG_HCI_OCF_READ_LOCAL_NAME:
616 case NG_HCI_OCF_READ_UNIT_CLASS:
617 case NG_HCI_OCF_WRITE_UNIT_CLASS:
618 case NG_HCI_OCF_READ_LE_HOST_SUPPORTED:
619 case NG_HCI_OCF_WRITE_LE_HOST_SUPPORTED:
620 /* These do not need post processing */
621 break;
622
623 case NG_HCI_OCF_RESET: {
624 ng_hci_unit_con_p con = NULL;
625 int size;
626
627 /*
628 * XXX
629 *
630 * After RESET command unit goes into standby mode
631 * and all operational state is lost. Host controller
632 * will revert to default values for all parameters.
633 *
634 * For now we shall terminate all connections and drop
635 * inited bit. After RESET unit must be re-initialized.
636 */
637
638 while (!LIST_EMPTY(&unit->con_list)) {
639 con = LIST_FIRST(&unit->con_list);
640
641 /* Remove all timeouts (if any) */
642 if (con->flags & NG_HCI_CON_TIMEOUT_PENDING)
643 ng_hci_con_untimeout(con);
644
645 /* Connection terminated by local host */
646 ng_hci_lp_discon_ind(con, 0x16);
647 ng_hci_free_con(con);
648 }
649
650 NG_HCI_BUFF_ACL_TOTAL(unit->buffer, size);
651 NG_HCI_BUFF_ACL_FREE(unit->buffer, size);
652
653 NG_HCI_BUFF_SCO_TOTAL(unit->buffer, size);
654 NG_HCI_BUFF_SCO_FREE(unit->buffer, size);
655
656 unit->state &= ~NG_HCI_UNIT_INITED;
657 } break;
658
659 default:
660 error = EINVAL;
661 break;
662 }
663
664 NG_FREE_M(mcp);
665 NG_FREE_M(mrp);
666
667 return (error);
668 } /* process_hc_baseband_params */
669
670 /*
671 * Process info command return parameters
672 */
673
674 static int
675 process_info_params(ng_hci_unit_p unit, u_int16_t ocf, struct mbuf *mcp,
676 struct mbuf *mrp)
677 {
678 int error = 0, len;
679
680 switch (ocf) {
681 case NG_HCI_OCF_READ_LOCAL_VER:
682 case NG_HCI_OCF_READ_COUNTRY_CODE:
683 break;
684
685 case NG_HCI_OCF_READ_LOCAL_FEATURES:
686 m_adj(mrp, sizeof(u_int8_t));
687 len = min(mrp->m_pkthdr.len, sizeof(unit->features));
688 m_copydata(mrp, 0, len, (caddr_t) unit->features);
689 break;
690
691 case NG_HCI_OCF_READ_BUFFER_SIZE: {
692 ng_hci_read_buffer_size_rp *rp = NULL;
693
694 /* Do not update buffer descriptor if node was initialized */
695 if ((unit->state & NG_HCI_UNIT_READY) == NG_HCI_UNIT_READY)
696 break;
697
698 NG_HCI_M_PULLUP(mrp, sizeof(*rp));
699 if (mrp != NULL) {
700 rp = mtod(mrp, ng_hci_read_buffer_size_rp *);
701
702 NG_HCI_BUFF_ACL_SET(
703 unit->buffer,
704 le16toh(rp->num_acl_pkt), /* number */
705 le16toh(rp->max_acl_size), /* size */
706 le16toh(rp->num_acl_pkt) /* free */
707 );
708
709 NG_HCI_BUFF_SCO_SET(
710 unit->buffer,
711 le16toh(rp->num_sco_pkt), /* number */
712 rp->max_sco_size, /* size */
713 le16toh(rp->num_sco_pkt) /* free */
714 );
715
716 /* Let upper layers know */
717 ng_hci_node_is_up(unit->node, unit->acl, NULL, 0);
718 ng_hci_node_is_up(unit->node, unit->sco, NULL, 0);
719 } else
720 error = ENOBUFS;
721 } break;
722
723 case NG_HCI_OCF_READ_BDADDR:
724 /* Do not update BD_ADDR if node was initialized */
725 if ((unit->state & NG_HCI_UNIT_READY) == NG_HCI_UNIT_READY)
726 break;
727
728 m_adj(mrp, sizeof(u_int8_t));
729 len = min(mrp->m_pkthdr.len, sizeof(unit->bdaddr));
730 m_copydata(mrp, 0, len, (caddr_t) &unit->bdaddr);
731
732 /* Let upper layers know */
733 ng_hci_node_is_up(unit->node, unit->acl, NULL, 0);
734 ng_hci_node_is_up(unit->node, unit->sco, NULL, 0);
735 break;
736
737 default:
738 error = EINVAL;
739 break;
740 }
741
742 NG_FREE_M(mcp);
743 NG_FREE_M(mrp);
744
745 return (error);
746 } /* process_info_params */
747
748 /*
749 * Process status command return parameters
750 */
751
752 static int
753 process_status_params(ng_hci_unit_p unit, u_int16_t ocf, struct mbuf *mcp,
754 struct mbuf *mrp)
755 {
756 int error = 0;
757
758 switch (ocf) {
759 case NG_HCI_OCF_READ_FAILED_CONTACT_CNTR:
760 case NG_HCI_OCF_RESET_FAILED_CONTACT_CNTR:
761 case NG_HCI_OCF_GET_LINK_QUALITY:
762 case NG_HCI_OCF_READ_RSSI:
763 /* These do not need post processing */
764 break;
765
766 default:
767 error = EINVAL;
768 break;
769 }
770
771 NG_FREE_M(mcp);
772 NG_FREE_M(mrp);
773
774 return (error);
775 } /* process_status_params */
776
777 /*
778 * Process testing command return parameters
779 */
780
781 int
782 process_testing_params(ng_hci_unit_p unit, u_int16_t ocf, struct mbuf *mcp,
783 struct mbuf *mrp)
784 {
785 int error = 0;
786
787 switch (ocf) {
788 /*
789 * XXX FIXME
790 * We do not support these features at this time. However,
791 * HCI node could support this and do something smart. At least
792 * node can change unit state.
793 */
794
795 case NG_HCI_OCF_READ_LOOPBACK_MODE:
796 case NG_HCI_OCF_WRITE_LOOPBACK_MODE:
797 case NG_HCI_OCF_ENABLE_UNIT_UNDER_TEST:
798 break;
799
800 default:
801 error = EINVAL;
802 break;
803 }
804
805 NG_FREE_M(mcp);
806 NG_FREE_M(mrp);
807
808 return (error);
809 } /* process_testing_params */
810
811 /*
812 * Process LE command return parameters
813 */
814
815 static int
816 process_le_params(ng_hci_unit_p unit, u_int16_t ocf,
817 struct mbuf *mcp, struct mbuf *mrp)
818 {
819 int error = 0;
820
821 switch (ocf){
822 case NG_HCI_OCF_LE_SET_EVENT_MASK:
823 case NG_HCI_OCF_LE_READ_BUFFER_SIZE:
824 case NG_HCI_OCF_LE_READ_LOCAL_SUPPORTED_FEATURES:
825 case NG_HCI_OCF_LE_SET_RANDOM_ADDRESS:
826 case NG_HCI_OCF_LE_SET_ADVERTISING_PARAMETERS:
827 case NG_HCI_OCF_LE_READ_ADVERTISING_CHANNEL_TX_POWER:
828 case NG_HCI_OCF_LE_SET_ADVERTISING_DATA:
829 case NG_HCI_OCF_LE_SET_SCAN_RESPONSE_DATA:
830 case NG_HCI_OCF_LE_SET_ADVERTISE_ENABLE:
831 case NG_HCI_OCF_LE_SET_SCAN_PARAMETERS:
832 case NG_HCI_OCF_LE_SET_SCAN_ENABLE:
833 case NG_HCI_OCF_LE_CREATE_CONNECTION_CANCEL:
834 case NG_HCI_OCF_LE_CLEAR_WHITE_LIST:
835 case NG_HCI_OCF_LE_READ_WHITE_LIST_SIZE:
836 case NG_HCI_OCF_LE_ADD_DEVICE_TO_WHITE_LIST:
837 case NG_HCI_OCF_LE_REMOVE_DEVICE_FROM_WHITE_LIST:
838 case NG_HCI_OCF_LE_SET_HOST_CHANNEL_CLASSIFICATION:
839 case NG_HCI_OCF_LE_READ_CHANNEL_MAP:
840 case NG_HCI_OCF_LE_ENCRYPT:
841 case NG_HCI_OCF_LE_RAND:
842 case NG_HCI_OCF_LE_LONG_TERM_KEY_REQUEST_REPLY:
843 case NG_HCI_OCF_LE_LONG_TERM_KEY_REQUEST_NEGATIVE_REPLY:
844 case NG_HCI_OCF_LE_READ_SUPPORTED_STATES:
845 case NG_HCI_OCF_LE_RECEIVER_TEST:
846 case NG_HCI_OCF_LE_TRANSMITTER_TEST:
847 case NG_HCI_OCF_LE_TEST_END:
848
849 /* These do not need post processing */
850 break;
851 case NG_HCI_OCF_LE_CREATE_CONNECTION:
852 case NG_HCI_OCF_LE_CONNECTION_UPDATE:
853 case NG_HCI_OCF_LE_READ_REMOTE_USED_FEATURES:
854 case NG_HCI_OCF_LE_START_ENCRYPTION:
855
856 default:
857 /*
858 * None of these command was supposed to generate
859 * Command_Complete event. Instead Command_Status event
860 * should have been generated and then appropriate event
861 * should have been sent to indicate the final result.
862 */
863
864 error = EINVAL;
865 break;
866 }
867
868 NG_FREE_M(mcp);
869 NG_FREE_M(mrp);
870
871 return (error);
872
873 }
874
875 static int
876 process_le_status(ng_hci_unit_p unit,ng_hci_command_status_ep *ep,
877 struct mbuf *mcp)
878 {
879 int error = 0;
880
881 switch (NG_HCI_OCF(ep->opcode)){
882 case NG_HCI_OCF_LE_CREATE_CONNECTION:
883 case NG_HCI_OCF_LE_CONNECTION_UPDATE:
884 case NG_HCI_OCF_LE_READ_REMOTE_USED_FEATURES:
885 case NG_HCI_OCF_LE_START_ENCRYPTION:
886
887 /* These do not need post processing */
888 break;
889
890 case NG_HCI_OCF_LE_SET_EVENT_MASK:
891 case NG_HCI_OCF_LE_READ_BUFFER_SIZE:
892 case NG_HCI_OCF_LE_READ_LOCAL_SUPPORTED_FEATURES:
893 case NG_HCI_OCF_LE_SET_RANDOM_ADDRESS:
894 case NG_HCI_OCF_LE_SET_ADVERTISING_PARAMETERS:
895 case NG_HCI_OCF_LE_READ_ADVERTISING_CHANNEL_TX_POWER:
896 case NG_HCI_OCF_LE_SET_ADVERTISING_DATA:
897 case NG_HCI_OCF_LE_SET_SCAN_RESPONSE_DATA:
898 case NG_HCI_OCF_LE_SET_ADVERTISE_ENABLE:
899 case NG_HCI_OCF_LE_SET_SCAN_PARAMETERS:
900 case NG_HCI_OCF_LE_SET_SCAN_ENABLE:
901 case NG_HCI_OCF_LE_CREATE_CONNECTION_CANCEL:
902 case NG_HCI_OCF_LE_CLEAR_WHITE_LIST:
903 case NG_HCI_OCF_LE_READ_WHITE_LIST_SIZE:
904 case NG_HCI_OCF_LE_ADD_DEVICE_TO_WHITE_LIST:
905 case NG_HCI_OCF_LE_REMOVE_DEVICE_FROM_WHITE_LIST:
906 case NG_HCI_OCF_LE_SET_HOST_CHANNEL_CLASSIFICATION:
907 case NG_HCI_OCF_LE_READ_CHANNEL_MAP:
908 case NG_HCI_OCF_LE_ENCRYPT:
909 case NG_HCI_OCF_LE_RAND:
910 case NG_HCI_OCF_LE_LONG_TERM_KEY_REQUEST_REPLY:
911 case NG_HCI_OCF_LE_LONG_TERM_KEY_REQUEST_NEGATIVE_REPLY:
912 case NG_HCI_OCF_LE_READ_SUPPORTED_STATES:
913 case NG_HCI_OCF_LE_RECEIVER_TEST:
914 case NG_HCI_OCF_LE_TRANSMITTER_TEST:
915 case NG_HCI_OCF_LE_TEST_END:
916
917 default:
918 /*
919 * None of these command was supposed to generate
920 * Command_Stutus event. Command Complete instead.
921 */
922
923 error = EINVAL;
924 break;
925 }
926
927 NG_FREE_M(mcp);
928
929 return (error);
930
931 }
932
933 /*
934 * Process link control command status
935 */
936
937 static int
938 process_link_control_status(ng_hci_unit_p unit, ng_hci_command_status_ep *ep,
939 struct mbuf *mcp)
940 {
941 int error = 0;
942
943 switch (NG_HCI_OCF(ep->opcode)) {
944 case NG_HCI_OCF_INQUIRY:
945 case NG_HCI_OCF_DISCON: /* XXX */
946 case NG_HCI_OCF_REJECT_CON: /* XXX */
947 case NG_HCI_OCF_CHANGE_CON_PKT_TYPE:
948 case NG_HCI_OCF_AUTH_REQ:
949 case NG_HCI_OCF_SET_CON_ENCRYPTION:
950 case NG_HCI_OCF_CHANGE_CON_LINK_KEY:
951 case NG_HCI_OCF_MASTER_LINK_KEY:
952 case NG_HCI_OCF_REMOTE_NAME_REQ:
953 case NG_HCI_OCF_READ_REMOTE_FEATURES:
954 case NG_HCI_OCF_READ_REMOTE_VER_INFO:
955 case NG_HCI_OCF_READ_CLOCK_OFFSET:
956 /* These do not need post processing */
957 break;
958
959 case NG_HCI_OCF_CREATE_CON:
960 break;
961
962 case NG_HCI_OCF_ADD_SCO_CON:
963 break;
964
965 case NG_HCI_OCF_ACCEPT_CON:
966 break;
967
968 case NG_HCI_OCF_INQUIRY_CANCEL:
969 case NG_HCI_OCF_PERIODIC_INQUIRY:
970 case NG_HCI_OCF_EXIT_PERIODIC_INQUIRY:
971 case NG_HCI_OCF_LINK_KEY_REP:
972 case NG_HCI_OCF_LINK_KEY_NEG_REP:
973 case NG_HCI_OCF_PIN_CODE_REP:
974 case NG_HCI_OCF_PIN_CODE_NEG_REP:
975 default:
976
977 /*
978 * None of these command was supposed to generate
979 * Command_Status event. Instead Command_Complete event
980 * should have been sent.
981 */
982
983 error = EINVAL;
984 break;
985 }
986
987 NG_FREE_M(mcp);
988
989 return (error);
990 } /* process_link_control_status */
991
992 /*
993 * Process link policy command status
994 */
995
996 static int
997 process_link_policy_status(ng_hci_unit_p unit, ng_hci_command_status_ep *ep,
998 struct mbuf *mcp)
999 {
1000 int error = 0;
1001
1002 switch (NG_HCI_OCF(ep->opcode)) {
1003 case NG_HCI_OCF_HOLD_MODE:
1004 case NG_HCI_OCF_SNIFF_MODE:
1005 case NG_HCI_OCF_EXIT_SNIFF_MODE:
1006 case NG_HCI_OCF_PARK_MODE:
1007 case NG_HCI_OCF_EXIT_PARK_MODE:
1008 case NG_HCI_OCF_SWITCH_ROLE:
1009 /* These do not need post processing */
1010 break;
1011
1012 case NG_HCI_OCF_QOS_SETUP:
1013 break;
1014
1015 case NG_HCI_OCF_ROLE_DISCOVERY:
1016 case NG_HCI_OCF_READ_LINK_POLICY_SETTINGS:
1017 case NG_HCI_OCF_WRITE_LINK_POLICY_SETTINGS:
1018 default:
1019
1020 /*
1021 * None of these command was supposed to generate
1022 * Command_Status event. Instead Command_Complete event
1023 * should have been sent.
1024 */
1025
1026 error = EINVAL;
1027 break;
1028 }
1029
1030 NG_FREE_M(mcp);
1031
1032 return (error);
1033 } /* process_link_policy_status */
Cache object: 93abc058ebcee1b8a043a2dddcd50039
|