xref: /freebsd/usr.sbin/bluetooth/hccontrol/host_controller_baseband.c (revision 70e0bbedef95258a4dadc996d641a9bebd3f107d)
1 /*
2  * host_controller_baseband.c
3  *
4  * Copyright (c) 2001-2002 Maksim Yevmenkin <m_evmenkin@yahoo.com>
5  * All rights reserved.
6  *
7  * Redistribution and use in source and binary forms, with or without
8  * modification, are permitted provided that the following conditions
9  * are met:
10  * 1. Redistributions of source code must retain the above copyright
11  *    notice, this list of conditions and the following disclaimer.
12  * 2. Redistributions in binary form must reproduce the above copyright
13  *    notice, this list of conditions and the following disclaimer in the
14  *    documentation and/or other materials provided with the distribution.
15  *
16  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
17  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
18  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
19  * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
20  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
21  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
22  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
23  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
24  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
25  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
26  * SUCH DAMAGE.
27  *
28  * $Id: host_controller_baseband.c,v 1.4 2003/08/18 19:19:53 max Exp $
29  * $FreeBSD$
30  */
31 
32 #include <bluetooth.h>
33 #include <errno.h>
34 #include <stdio.h>
35 #include <string.h>
36 #include "hccontrol.h"
37 
38 /* Convert hex ASCII to int4 */
39 static int
40 hci_hexa2int4(const char *a)
41 {
42 	if ('0' <= *a && *a <= '9')
43 		return (*a - '0');
44 
45 	if ('A' <= *a && *a <= 'F')
46 		return (*a - 'A' + 0xa);
47 
48 	if ('a' <= *a && *a <= 'f')
49 		return (*a - 'a' + 0xa);
50 
51 	return (-1);
52 }
53 
54 /* Convert hex ASCII to int8 */
55 static int
56 hci_hexa2int8(const char *a)
57 {
58 	int	hi = hci_hexa2int4(a);
59 	int	lo = hci_hexa2int4(a + 1);
60 
61 	if (hi < 0 || lo < 0)
62 		return (-1);
63 
64 	return ((hi << 4) | lo);
65 }
66 
67 /* Convert ascii hex string to the uint8_t[] */
68 static int
69 hci_hexstring2array(char const *s, uint8_t *a, int asize)
70 {
71 	int	i, l, b;
72 
73 	l = strlen(s) / 2;
74 	if (l > asize)
75 		l = asize;
76 
77 	for (i = 0; i < l; i++) {
78 		b = hci_hexa2int8(s + i * 2);
79 		if (b < 0)
80 			return (-1);
81 
82 		a[i] = (b & 0xff);
83 	}
84 
85 	return (0);
86 }
87 
88 /* Send RESET to the unit */
89 static int
90 hci_reset(int s, int argc, char **argv)
91 {
92 	ng_hci_status_rp	rp;
93 	int			n;
94 
95 	n = sizeof(rp);
96 	if (hci_simple_request(s, NG_HCI_OPCODE(NG_HCI_OGF_HC_BASEBAND,
97 			NG_HCI_OCF_RESET), (char *) &rp, &n) == ERROR)
98 		return (ERROR);
99 
100 	if (rp.status != 0x00) {
101 		fprintf(stdout, "Status: %s [%#02x]\n",
102 			hci_status2str(rp.status), rp.status);
103 		return (FAILED);
104 	}
105 
106 	return (OK);
107 } /* hci_reset */
108 
109 /* Send Read_PIN_Type command to the unit */
110 static int
111 hci_read_pin_type(int s, int argc, char **argv)
112 {
113 	ng_hci_read_pin_type_rp	rp;
114 	int			n;
115 
116 	n = sizeof(rp);
117 	if (hci_simple_request(s, NG_HCI_OPCODE(NG_HCI_OGF_HC_BASEBAND,
118 			NG_HCI_OCF_READ_PIN_TYPE),
119 			(char *) &rp, &n) == ERROR)
120 		return (ERROR);
121 
122 	if (rp.status != 0x00) {
123 		fprintf(stdout, "Status: %s [%#02x]\n",
124 			hci_status2str(rp.status), rp.status);
125 		return (FAILED);
126 	}
127 
128 	fprintf(stdout, "PIN type: %s [%#02x]\n",
129 			hci_pin2str(rp.pin_type), rp.pin_type);
130 
131 	return (OK);
132 } /* hci_read_pin_type */
133 
134 /* Send Write_PIN_Type command to the unit */
135 static int
136 hci_write_pin_type(int s, int argc, char **argv)
137 {
138 	ng_hci_write_pin_type_cp	cp;
139 	ng_hci_write_pin_type_rp	rp;
140 	int				n;
141 
142 	/* parse command parameters */
143 	switch (argc) {
144 	case 1:
145 		if (sscanf(argv[0], "%d", &n) != 1 || n < 0 || n > 1)
146 			return (USAGE);
147 
148 		cp.pin_type = (uint8_t) n;
149 		break;
150 
151 	default:
152 		return (USAGE);
153 	}
154 
155 	/* send command */
156 	n = sizeof(rp);
157 	if (hci_request(s, NG_HCI_OPCODE(NG_HCI_OGF_HC_BASEBAND,
158 			NG_HCI_OCF_WRITE_PIN_TYPE),
159 			(char const *) &cp, sizeof(cp),
160 			(char *) &rp , &n) ==  ERROR)
161 		return (ERROR);
162 
163 	if (rp.status != 0x00) {
164 		fprintf(stdout, "Status: %s [%#02x]\n",
165 			hci_status2str(rp.status), rp.status);
166 		return (FAILED);
167 	}
168 
169 	return (OK);
170 } /* hci_write_pin_type */
171 
172 /* Send Read_Stored_Link_Key command to the unit */
173 static int
174 hci_read_stored_link_key(int s, int argc, char **argv)
175 {
176 	struct {
177 		ng_hci_cmd_pkt_t			hdr;
178 		ng_hci_read_stored_link_key_cp		cp;
179 	} __attribute__ ((packed))			cmd;
180 
181 	struct {
182 		ng_hci_event_pkt_t			hdr;
183 		union {
184 			ng_hci_command_compl_ep		cc;
185 			ng_hci_return_link_keys_ep	key;
186 			uint8_t				b[NG_HCI_EVENT_PKT_SIZE];
187 		}					ep;
188 	} __attribute__ ((packed))			event;
189 
190 	int						n, n1;
191 
192 	/* Send command */
193 	memset(&cmd, 0, sizeof(cmd));
194 	cmd.hdr.type = NG_HCI_CMD_PKT;
195 	cmd.hdr.opcode = htole16(NG_HCI_OPCODE(NG_HCI_OGF_HC_BASEBAND,
196 				NG_HCI_OCF_READ_STORED_LINK_KEY));
197 	cmd.hdr.length = sizeof(cmd.cp);
198 
199 	switch (argc) {
200 	case 1:
201 		/* parse BD_ADDR */
202 		if (!bt_aton(argv[0], &cmd.cp.bdaddr)) {
203 			struct hostent	*he = NULL;
204 
205 			if ((he = bt_gethostbyname(argv[0])) == NULL)
206 				return (USAGE);
207 
208 			memcpy(&cmd.cp.bdaddr, he->h_addr, sizeof(cmd.cp.bdaddr));
209 		}
210 		break;
211 
212 	default:
213 		cmd.cp.read_all = 1;
214 		break;
215 	}
216 
217 	if (hci_send(s, (char const *) &cmd, sizeof(cmd)) != OK)
218 		return (ERROR);
219 
220 	/* Receive events */
221 again:
222 	memset(&event, 0, sizeof(event));
223 	n = sizeof(event);
224 	if (hci_recv(s, (char *) &event, &n) != OK)
225 		return (ERROR);
226 
227 	if (n <= sizeof(event.hdr)) {
228 		errno = EMSGSIZE;
229 		return (ERROR);
230 	}
231 
232 	if (event.hdr.type != NG_HCI_EVENT_PKT) {
233 		errno = EIO;
234 		return (ERROR);
235 	}
236 
237 	/* Parse event */
238 	switch (event.hdr.event) {
239 	case NG_HCI_EVENT_COMMAND_COMPL: {
240 		ng_hci_read_stored_link_key_rp	*rp = NULL;
241 
242 		if (event.ep.cc.opcode == 0x0000 ||
243 		    event.ep.cc.opcode != cmd.hdr.opcode)
244 			goto again;
245 
246 		rp = (ng_hci_read_stored_link_key_rp *)(event.ep.b +
247 				sizeof(event.ep.cc));
248 
249 		fprintf(stdout, "Complete: Status: %s [%#x]\n",
250 				hci_status2str(rp->status), rp->status);
251 		fprintf(stdout, "Maximum Number of keys: %d\n",
252 				le16toh(rp->max_num_keys));
253 		fprintf(stdout, "Number of keys read: %d\n",
254 				le16toh(rp->num_keys_read));
255 		} break;
256 
257 	case NG_HCI_EVENT_RETURN_LINK_KEYS: {
258 		struct _key {
259 			bdaddr_t	bdaddr;
260 			uint8_t		key[NG_HCI_KEY_SIZE];
261 		} __attribute__ ((packed))	*k = NULL;
262 
263 		fprintf(stdout, "Event: Number of keys: %d\n",
264 			event.ep.key.num_keys);
265 
266 		k = (struct _key *)(event.ep.b + sizeof(event.ep.key));
267 		for (n = 0; n < event.ep.key.num_keys; n++) {
268 			fprintf(stdout, "\t%d: %s ",
269 				n + 1, hci_bdaddr2str(&k->bdaddr));
270 
271 			for (n1 = 0; n1 < sizeof(k->key); n1++)
272 				fprintf(stdout, "%02x", k->key[n1]);
273 			fprintf(stdout, "\n");
274 
275 			k ++;
276 		}
277 
278 		goto again;
279 
280 		} break;
281 
282 	default:
283 		goto again;
284 	}
285 
286 	return (OK);
287 } /* hci_read_store_link_key */
288 
289 /* Send Write_Stored_Link_Key command to the unit */
290 static int
291 hci_write_stored_link_key(int s, int argc, char **argv)
292 {
293 	struct {
294 		ng_hci_write_stored_link_key_cp	p;
295 		bdaddr_t			bdaddr;
296 		uint8_t				key[NG_HCI_KEY_SIZE];
297 	}					cp;
298 	ng_hci_write_stored_link_key_rp		rp;
299 	int32_t					n;
300 
301 	memset(&cp, 0, sizeof(cp));
302 
303 	switch (argc) {
304 	case 2:
305 		cp.p.num_keys_write = 1;
306 
307 		/* parse BD_ADDR */
308 		if (!bt_aton(argv[0], &cp.bdaddr)) {
309 			struct hostent	*he = NULL;
310 
311 			if ((he = bt_gethostbyname(argv[0])) == NULL)
312 				return (USAGE);
313 
314 			memcpy(&cp.bdaddr, he->h_addr, sizeof(cp.bdaddr));
315 		}
316 
317 		/* parse key */
318 		if (hci_hexstring2array(argv[1], cp.key, sizeof(cp.key)) < 0)
319 			return (USAGE);
320 		break;
321 
322 	default:
323 		return (USAGE);
324 	}
325 
326 	/* send command */
327 	n = sizeof(rp);
328 	if (hci_request(s, NG_HCI_OPCODE(NG_HCI_OGF_HC_BASEBAND,
329 			NG_HCI_OCF_WRITE_STORED_LINK_KEY),
330 			(char const *) &cp, sizeof(cp),
331 			(char *) &rp, &n) == ERROR)
332 		return (ERROR);
333 
334 	if (rp.status != 0x00) {
335 		fprintf(stdout, "Status: %s [%#02x]\n",
336 			hci_status2str(rp.status), rp.status);
337 		return (FAILED);
338 	}
339 
340 	fprintf(stdout, "Number of keys written: %d\n", rp.num_keys_written);
341 
342 	return (OK);
343 } /* hci_write_stored_link_key */
344 
345 
346 /* Send Delete_Stored_Link_Key command to the unit */
347 static int
348 hci_delete_stored_link_key(int s, int argc, char **argv)
349 {
350 	ng_hci_delete_stored_link_key_cp	cp;
351 	ng_hci_delete_stored_link_key_rp	rp;
352 	int32_t					n;
353 
354 	memset(&cp, 0, sizeof(cp));
355 
356 	switch (argc) {
357 	case 1:
358 		/* parse BD_ADDR */
359 		if (!bt_aton(argv[0], &cp.bdaddr)) {
360 			struct hostent	*he = NULL;
361 
362 			if ((he = bt_gethostbyname(argv[0])) == NULL)
363 				return (USAGE);
364 
365 			memcpy(&cp.bdaddr, he->h_addr, sizeof(cp.bdaddr));
366 		}
367 		break;
368 
369 	default:
370 		cp.delete_all = 1;
371 		break;
372 	}
373 
374 	/* send command */
375 	n = sizeof(cp);
376 	if (hci_request(s, NG_HCI_OPCODE(NG_HCI_OGF_HC_BASEBAND,
377 			NG_HCI_OCF_DELETE_STORED_LINK_KEY),
378 			(char const *) &cp, sizeof(cp),
379 			(char *) &rp, &n) == ERROR)
380 		return (ERROR);
381 
382 	if (rp.status != 0x00) {
383 		fprintf(stdout, "Status: %s [%#02x]\n",
384 			hci_status2str(rp.status), rp.status);
385 		return (FAILED);
386 	}
387 
388 	fprintf(stdout, "Number of keys deleted: %d\n", rp.num_keys_deleted);
389 
390 	return (OK);
391 } /* hci_delete_stored_link_key */
392 
393 /* Send Change_Local_Name command to the unit */
394 static int
395 hci_change_local_name(int s, int argc, char **argv)
396 {
397 	ng_hci_change_local_name_cp	cp;
398 	ng_hci_change_local_name_rp	rp;
399 	int				n;
400 
401 	/* parse command parameters */
402 	switch (argc) {
403 	case 1:
404 		snprintf(cp.name, sizeof(cp.name), "%s", argv[0]);
405 		break;
406 
407 	default:
408 		return (USAGE);
409 	}
410 
411 	/* send command */
412 	n = sizeof(rp);
413 	if (hci_request(s, NG_HCI_OPCODE(NG_HCI_OGF_HC_BASEBAND,
414 			NG_HCI_OCF_CHANGE_LOCAL_NAME),
415 			(char const *) &cp, sizeof(cp),
416 			(char *) &rp, &n) == ERROR)
417 		return (ERROR);
418 
419 	if (rp.status != 0x00) {
420 		fprintf(stdout, "Status: %s [%#02x]\n",
421 			hci_status2str(rp.status), rp.status);
422 		return (FAILED);
423 	}
424 
425 	return (OK);
426 } /* hci_change_local_name */
427 
428 /* Send Read_Local_Name command to the unit */
429 static int
430 hci_read_local_name(int s, int argc, char **argv)
431 {
432 	ng_hci_read_local_name_rp	rp;
433 	int				n;
434 
435 	n = sizeof(rp);
436 	if (hci_simple_request(s, NG_HCI_OPCODE(NG_HCI_OGF_HC_BASEBAND,
437 			NG_HCI_OCF_READ_LOCAL_NAME),
438 			(char *) &rp, &n) == ERROR)
439 		return (ERROR);
440 
441 	if (rp.status != 0x00) {
442 		fprintf(stdout, "Status: %s [%#02x]\n",
443 			hci_status2str(rp.status), rp.status);
444 		return (FAILED);
445 	}
446 
447 	fprintf(stdout, "Local name: %s\n", rp.name);
448 
449 	return (OK);
450 } /* hci_read_local_name */
451 
452 /* Send Read_Connection_Accept_Timeout to the unit */
453 static int
454 hci_read_connection_accept_timeout(int s, int argc, char **argv)
455 {
456 	ng_hci_read_con_accept_timo_rp	rp;
457 	int				n;
458 
459 	n = sizeof(rp);
460 	if (hci_simple_request(s, NG_HCI_OPCODE(NG_HCI_OGF_HC_BASEBAND,
461 			NG_HCI_OCF_READ_CON_ACCEPT_TIMO),
462 			(char *) &rp, &n) == ERROR)
463 		return (ERROR);
464 
465 	if (rp.status != 0x00) {
466 		fprintf(stdout, "Status: %s [%#02x]\n",
467 			hci_status2str(rp.status), rp.status);
468 		return (FAILED);
469 	}
470 
471 	rp.timeout = le16toh(rp.timeout);
472 	fprintf(stdout, "Connection accept timeout: %.2f msec [%d slots]\n",
473 			rp.timeout * 0.625, rp.timeout);
474 
475 	return (OK);
476 } /* hci_read_connection_accept_timeout */
477 
478 /* Send Write_Connection_Accept_Timeout to the unit */
479 static int
480 hci_write_connection_accept_timeout(int s, int argc, char **argv)
481 {
482 	ng_hci_write_con_accept_timo_cp	cp;
483 	ng_hci_write_con_accept_timo_rp	rp;
484 	int				n;
485 
486 	/* parse command parameters */
487 	switch (argc) {
488 	case 1:
489 		if (sscanf(argv[0], "%d", &n) != 1 || n < 1 || n > 0xb540)
490 			return (USAGE);
491 
492 		cp.timeout = (uint16_t) n;
493 		cp.timeout = htole16(cp.timeout);
494 		break;
495 
496 	default:
497 		return (USAGE);
498 	}
499 
500 	/* send command */
501 	n = sizeof(rp);
502 	if (hci_request(s, NG_HCI_OPCODE(NG_HCI_OGF_HC_BASEBAND,
503 			NG_HCI_OCF_WRITE_CON_ACCEPT_TIMO),
504 			(char const *) &cp, sizeof(cp),
505 			(char *) &rp, &n) == ERROR)
506 		return (ERROR);
507 
508 	if (rp.status != 0x00) {
509 		fprintf(stdout, "Status: %s [%#02x]\n",
510 			hci_status2str(rp.status), rp.status);
511 		return (FAILED);
512 	}
513 
514 	return (OK);
515 } /* hci_write_connection_accept_timeout */
516 
517 /* Send Read_Page_Timeout command to the unit */
518 static int
519 hci_read_page_timeout(int s, int argc, char **argv)
520 {
521 	ng_hci_read_page_timo_rp	rp;
522 	int				n;
523 
524 	n = sizeof(rp);
525 	if (hci_simple_request(s, NG_HCI_OPCODE(NG_HCI_OGF_HC_BASEBAND,
526 			NG_HCI_OCF_READ_PAGE_TIMO),
527 			(char *) &rp, &n) == ERROR)
528 		return (ERROR);
529 
530 	if (rp.status != 0x00) {
531 		fprintf(stdout, "Status: %s [%#02x]\n",
532 			hci_status2str(rp.status), rp.status);
533 		return (FAILED);
534 	}
535 
536 	rp.timeout = le16toh(rp.timeout);
537 	fprintf(stdout, "Page timeout: %.2f msec [%d slots]\n",
538 		rp.timeout * 0.625, rp.timeout);
539 
540 	return (OK);
541 } /* hci_read_page_timeoout */
542 
543 /* Send Write_Page_Timeout command to the unit */
544 static int
545 hci_write_page_timeout(int s, int argc, char **argv)
546 {
547 	ng_hci_write_page_timo_cp	cp;
548 	ng_hci_write_page_timo_rp	rp;
549 	int				n;
550 
551 	/* parse command parameters */
552 	switch (argc) {
553 	case 1:
554 		if (sscanf(argv[0], "%d", &n) != 1 || n < 1 || n > 0xffff)
555 			return (USAGE);
556 
557 		cp.timeout = (uint16_t) n;
558 		cp.timeout = htole16(cp.timeout);
559 		break;
560 
561 	default:
562 		return (USAGE);
563 	}
564 
565 	/* send command */
566 	n = sizeof(rp);
567 	if (hci_request(s, NG_HCI_OPCODE(NG_HCI_OGF_HC_BASEBAND,
568 			NG_HCI_OCF_WRITE_PAGE_TIMO),
569 			(char const *) &cp, sizeof(cp),
570 			(char *) &rp, &n) == ERROR)
571 		return (ERROR);
572 
573 	if (rp.status != 0x00) {
574 		fprintf(stdout, "Status: %s [%#02x]\n",
575 			hci_status2str(rp.status), rp.status);
576 		return (FAILED);
577 	}
578 
579 	return (OK);
580 } /* hci_write_page_timeout */
581 
582 /* Send Read_Scan_Enable command to the unit */
583 static int
584 hci_read_scan_enable(int s, int argc, char **argv)
585 {
586 	ng_hci_read_scan_enable_rp	rp;
587 	int				n;
588 
589 	n = sizeof(rp);
590 	if (hci_simple_request(s, NG_HCI_OPCODE(NG_HCI_OGF_HC_BASEBAND,
591 			NG_HCI_OCF_READ_SCAN_ENABLE),
592 			(char *) &rp, &n) == ERROR)
593 		return (ERROR);
594 
595 	if (rp.status != 0x00) {
596 		fprintf(stdout, "Status: %s [%#02x]\n",
597 			hci_status2str(rp.status), rp.status);
598 		return (FAILED);
599 	}
600 
601 	fprintf(stdout, "Scan enable: %s [%#02x]\n",
602 		hci_scan2str(rp.scan_enable), rp.scan_enable);
603 
604 	return (OK);
605 } /* hci_read_scan_enable */
606 
607 /* Send Write_Scan_Enable command to the unit */
608 static int
609 hci_write_scan_enable(int s, int argc, char **argv)
610 {
611 	ng_hci_write_scan_enable_cp	cp;
612 	ng_hci_write_scan_enable_rp	rp;
613 	int				n;
614 
615 	/* parse command parameters */
616 	switch (argc) {
617 	case 1:
618 		if (sscanf(argv[0], "%d", &n) != 1 || n < 0 || n > 3)
619 			return (USAGE);
620 
621 		cp.scan_enable = (uint8_t) n;
622 		break;
623 
624 	default:
625 		return (USAGE);
626 	}
627 
628 	n = sizeof(rp);
629 	if (hci_request(s, NG_HCI_OPCODE(NG_HCI_OGF_HC_BASEBAND,
630 			NG_HCI_OCF_WRITE_SCAN_ENABLE),
631 			(char const *) &cp, sizeof(cp),
632 			(char *) &rp, &n) == ERROR)
633 		return (ERROR);
634 
635 	if (rp.status != 0x00) {
636 		fprintf(stdout, "Status: %s [%#02x]\n",
637 			hci_status2str(rp.status), rp.status);
638 		return (FAILED);
639 	}
640 
641 	return (OK);
642 } /* hci_write_scan_enable */
643 
644 /* Send Read_Page_Scan_Activity command to the unit */
645 static int
646 hci_read_page_scan_activity(int s, int argc, char **argv)
647 {
648 	ng_hci_read_page_scan_activity_rp	rp;
649 	int					n;
650 
651 	n = sizeof(rp);
652 	if (hci_simple_request(s, NG_HCI_OPCODE(NG_HCI_OGF_HC_BASEBAND,
653 			NG_HCI_OCF_READ_PAGE_SCAN_ACTIVITY),
654 			(char *) &rp, &n) == ERROR)
655 		return (ERROR);
656 
657 	if (rp.status != 0x00) {
658 		fprintf(stdout, "Status: %s [%#02x]\n",
659 			hci_status2str(rp.status), rp.status);
660 		return (FAILED);
661 	}
662 
663 	rp.page_scan_interval = le16toh(rp.page_scan_interval);
664 	rp.page_scan_window = le16toh(rp.page_scan_window);
665 
666 	fprintf(stdout, "Page Scan Interval: %.2f msec [%d slots]\n",
667 		rp.page_scan_interval * 0.625, rp.page_scan_interval);
668 	fprintf(stdout, "Page Scan Window: %.2f msec [%d slots]\n",
669 		rp.page_scan_window * 0.625, rp.page_scan_window);
670 
671 	return (OK);
672 } /* hci_read_page_scan_activity */
673 
674 /* Send Write_Page_Scan_Activity command to the unit */
675 static int
676 hci_write_page_scan_activity(int s, int argc, char **argv)
677 {
678 	ng_hci_write_page_scan_activity_cp	cp;
679 	ng_hci_write_page_scan_activity_rp	rp;
680 	int					n;
681 
682 	/* parse command parameters */
683 	switch (argc) {
684 	case 2:
685 		/* page scan interval */
686 		if (sscanf(argv[0], "%d", &n) != 1 || n < 0x12 || n > 0x1000)
687 			return (USAGE);
688 
689 		cp.page_scan_interval = (uint16_t) n;
690 
691 		/* page scan window */
692 		if (sscanf(argv[1], "%d", &n) != 1 || n < 0x12 || n > 0x1000)
693 			return (USAGE);
694 
695 		cp.page_scan_window = (uint16_t) n;
696 
697 		if (cp.page_scan_window > cp.page_scan_interval)
698 			return (USAGE);
699 
700 		cp.page_scan_interval = htole16(cp.page_scan_interval);
701 		cp.page_scan_window = htole16(cp.page_scan_window);
702 		break;
703 
704 	default:
705 		return (USAGE);
706 	}
707 
708 	/* send command */
709 	n = sizeof(rp);
710 	if (hci_request(s, NG_HCI_OPCODE(NG_HCI_OGF_HC_BASEBAND,
711 			NG_HCI_OCF_WRITE_PAGE_SCAN_ACTIVITY),
712 			(char const *) &cp, sizeof(cp),
713 			(char *) &rp, &n) == ERROR)
714 		return (ERROR);
715 
716 	if (rp.status != 0x00) {
717 		fprintf(stdout, "Status: %s [%#02x]\n",
718 			hci_status2str(rp.status), rp.status);
719 		return (FAILED);
720 	}
721 
722 	return (OK);
723 } /* hci_write_page_scan_activity */
724 
725 /* Send Read_Inquiry_Scan_Activity command to the unit */
726 static int
727 hci_read_inquiry_scan_activity(int s, int argc, char **argv)
728 {
729 	ng_hci_read_inquiry_scan_activity_rp	rp;
730 	int					n;
731 
732 	n = sizeof(rp);
733 	if (hci_simple_request(s, NG_HCI_OPCODE(NG_HCI_OGF_HC_BASEBAND,
734 			NG_HCI_OCF_READ_INQUIRY_SCAN_ACTIVITY),
735 			(char *) &rp, &n) == ERROR)
736 		return (ERROR);
737 
738 	if (rp.status != 0x00) {
739 		fprintf(stdout, "Status: %s [%#02x]\n",
740 			hci_status2str(rp.status), rp.status);
741 		return (FAILED);
742 	}
743 
744 	rp.inquiry_scan_interval = le16toh(rp.inquiry_scan_interval);
745 	rp.inquiry_scan_window = le16toh(rp.inquiry_scan_window);
746 
747 	fprintf(stdout, "Inquiry Scan Interval: %.2f msec [%d slots]\n",
748 		rp.inquiry_scan_interval * 0.625, rp.inquiry_scan_interval);
749 	fprintf(stdout, "Inquiry Scan Window: %.2f msec [%d slots]\n",
750 		rp.inquiry_scan_window * 0.625, rp.inquiry_scan_interval);
751 
752 	return (OK);
753 } /* hci_read_inquiry_scan_activity */
754 
755 /* Send Write_Inquiry_Scan_Activity command to the unit */
756 static int
757 hci_write_inquiry_scan_activity(int s, int argc, char **argv)
758 {
759 	ng_hci_write_inquiry_scan_activity_cp	cp;
760 	ng_hci_write_inquiry_scan_activity_rp	rp;
761 	int					n;
762 
763 	/* parse command parameters */
764 	switch (argc) {
765 	case 2:
766 		/* inquiry scan interval */
767 		if (sscanf(argv[0], "%d", &n) != 1 || n < 0x12 || n > 0x1000)
768 			return (USAGE);
769 
770 		cp.inquiry_scan_interval = (uint16_t) n;
771 
772 		/* inquiry scan window */
773 		if (sscanf(argv[1], "%d", &n) != 1 || n < 0x12 || n > 0x1000)
774 			return (USAGE);
775 
776 		cp.inquiry_scan_window = (uint16_t) n;
777 
778 		if (cp.inquiry_scan_window > cp.inquiry_scan_interval)
779 			return (USAGE);
780 
781 		cp.inquiry_scan_interval =
782 			htole16(cp.inquiry_scan_interval);
783 		cp.inquiry_scan_window = htole16(cp.inquiry_scan_window);
784 		break;
785 
786 	default:
787 		return (USAGE);
788 	}
789 
790 	/* send command */
791 	n = sizeof(rp);
792 	if (hci_request(s, NG_HCI_OPCODE(NG_HCI_OGF_HC_BASEBAND,
793 			NG_HCI_OCF_WRITE_INQUIRY_SCAN_ACTIVITY),
794 			(char const *) &cp, sizeof(cp),
795 			(char *) &rp, &n) == ERROR)
796 		return (ERROR);
797 
798 	if (rp.status != 0x00) {
799 		fprintf(stdout, "Status: %s [%#02x]\n",
800 			hci_status2str(rp.status), rp.status);
801 		return (FAILED);
802 	}
803 
804 	return (OK);
805 } /* hci_write_inquiry_scan_activity */
806 
807 /* Send Read_Authentication_Enable command to the unit */
808 static int
809 hci_read_authentication_enable(int s, int argc, char **argv)
810 {
811 	ng_hci_read_auth_enable_rp	rp;
812 	int				n;
813 
814 	n = sizeof(rp);
815 	if (hci_simple_request(s, NG_HCI_OPCODE(NG_HCI_OGF_HC_BASEBAND,
816 			NG_HCI_OCF_READ_AUTH_ENABLE),
817 			(char *) &rp, &n) == ERROR)
818 		return (ERROR);
819 
820 	if (rp.status != 0x00) {
821 		fprintf(stdout, "Status: %s [%#02x]\n",
822 			hci_status2str(rp.status), rp.status);
823 		return (FAILED);
824 	}
825 
826 	fprintf(stdout, "Authentication Enable: %s [%d]\n",
827 		rp.auth_enable? "Enabled" : "Disabled", rp.auth_enable);
828 
829 	return (OK);
830 } /* hci_read_authentication_enable */
831 
832 /* Send Write_Authentication_Enable command to the unit */
833 static int
834 hci_write_authentication_enable(int s, int argc, char **argv)
835 {
836 	ng_hci_write_auth_enable_cp	cp;
837 	ng_hci_write_auth_enable_rp	rp;
838 	int				n;
839 
840 	/* parse command parameters */
841 	switch (argc) {
842 	case 1:
843 		if (sscanf(argv[0], "%d", &n) != 1 || n < 0 || n > 1)
844 			return (USAGE);
845 
846 		cp.auth_enable = (uint8_t) n;
847 		break;
848 
849 	default:
850 		return (USAGE);
851 	}
852 
853 	/* send command */
854 	n = sizeof(rp);
855 	if (hci_request(s, NG_HCI_OPCODE(NG_HCI_OGF_HC_BASEBAND,
856 			NG_HCI_OCF_WRITE_AUTH_ENABLE),
857 			(char const *) &cp, sizeof(cp),
858 			(char *) &rp, &n) == ERROR)
859 		return (ERROR);
860 
861 	if (rp.status != 0x00) {
862 		fprintf(stdout, "Status: %s [%#02x]\n",
863 			hci_status2str(rp.status), rp.status);
864 		return (FAILED);
865 	}
866 
867 	return (OK);
868 } /* hci_write_authentication_enable */
869 
870 /* Send Read_Encryption_Mode command to the unit */
871 static int
872 hci_read_encryption_mode(int s, int argc, char **argv)
873 {
874 	ng_hci_read_encryption_mode_rp	rp;
875 	int				n;
876 
877 	n = sizeof(rp);
878 	if (hci_simple_request(s, NG_HCI_OPCODE(NG_HCI_OGF_HC_BASEBAND,
879 			NG_HCI_OCF_READ_ENCRYPTION_MODE),
880 			(char *) &rp, &n) == ERROR)
881 		return (ERROR);
882 
883 	if (rp.status != 0x00) {
884 		fprintf(stdout, "Status: %s [%#02x]\n",
885 			hci_status2str(rp.status), rp.status);
886 		return (FAILED);
887 	}
888 
889 	fprintf(stdout, "Encryption mode: %s [%#02x]\n",
890 		hci_encrypt2str(rp.encryption_mode, 0), rp.encryption_mode);
891 
892 	return (OK);
893 } /* hci_read_encryption_mode */
894 
895 /* Send Write_Encryption_Mode command to the unit */
896 static int
897 hci_write_encryption_mode(int s, int argc, char **argv)
898 {
899 	ng_hci_write_encryption_mode_cp	cp;
900 	ng_hci_write_encryption_mode_rp	rp;
901 	int				n;
902 
903 	/* parse command parameters */
904 	switch (argc) {
905 	case 1:
906 		if (sscanf(argv[0], "%d", &n) != 1 || n < 0 || n > 2)
907 			return (USAGE);
908 
909 		cp.encryption_mode = (uint8_t) n;
910 		break;
911 
912 	default:
913 		return (USAGE);
914 	}
915 
916 	/* send command */
917 	n = sizeof(rp);
918 	if (hci_request(s, NG_HCI_OPCODE(NG_HCI_OGF_HC_BASEBAND,
919 			NG_HCI_OCF_WRITE_ENCRYPTION_MODE),
920 			(char const *) &cp, sizeof(cp),
921 			(char *) &rp, &n) == ERROR)
922 		return (ERROR);
923 
924 	if (rp.status != 0x00) {
925 		fprintf(stdout, "Status: %s [%#02x]\n",
926 			hci_status2str(rp.status), rp.status);
927 		return (FAILED);
928 	}
929 
930 	return (OK);
931 } /* hci_write_encryption_mode */
932 
933 /* Send Read_Class_Of_Device command to the unit */
934 static int
935 hci_read_class_of_device(int s, int argc, char **argv)
936 {
937 	ng_hci_read_unit_class_rp	rp;
938 	int				n;
939 
940 	n = sizeof(rp);
941 	if (hci_simple_request(s, NG_HCI_OPCODE(NG_HCI_OGF_HC_BASEBAND,
942 			NG_HCI_OCF_READ_UNIT_CLASS),
943 			(char *) &rp, &n) == ERROR)
944 		return (ERROR);
945 
946 	if (rp.status != 0x00) {
947 		fprintf(stdout, "Status: %s [%#02x]\n",
948 			hci_status2str(rp.status), rp.status);
949 		return (FAILED);
950 	}
951 
952 	fprintf(stdout, "Class: %02x:%02x:%02x\n",
953 		rp.uclass[2], rp.uclass[1], rp.uclass[0]);
954 
955 	return (0);
956 } /* hci_read_class_of_device */
957 
958 /* Send Write_Class_Of_Device command to the unit */
959 static int
960 hci_write_class_of_device(int s, int argc, char **argv)
961 {
962 	ng_hci_write_unit_class_cp	cp;
963 	ng_hci_write_unit_class_rp	rp;
964 	int				n0, n1, n2;
965 
966 	/* parse command parameters */
967 	switch (argc) {
968 	case 1:
969 		if (sscanf(argv[0], "%x:%x:%x", &n2, &n1, &n0) != 3)
970 			return (USAGE);
971 
972 		cp.uclass[0] = (n0 & 0xff);
973 		cp.uclass[1] = (n1 & 0xff);
974 		cp.uclass[2] = (n2 & 0xff);
975 		break;
976 
977 	default:
978 		return (USAGE);
979 	}
980 
981 	/* send command */
982 	n0 = sizeof(rp);
983 	if (hci_request(s, NG_HCI_OPCODE(NG_HCI_OGF_HC_BASEBAND,
984 			NG_HCI_OCF_WRITE_UNIT_CLASS),
985 			(char const *) &cp, sizeof(cp),
986 			(char *) &rp, &n0) == ERROR)
987 		return (ERROR);
988 
989 	if (rp.status != 0x00) {
990 		fprintf(stdout, "Status: %s [%#02x]\n",
991 			hci_status2str(rp.status), rp.status);
992 		return (FAILED);
993 	}
994 
995 	return (OK);
996 } /* hci_write_class_of_device */
997 
998 /* Send Read_Voice_Settings command to the unit */
999 static int
1000 hci_read_voice_settings(int s, int argc, char **argv)
1001 {
1002 	ng_hci_read_voice_settings_rp	rp;
1003 	int				n,
1004 					input_coding,
1005 					input_data_format,
1006 					input_sample_size;
1007 
1008 	n = sizeof(rp);
1009 	if (hci_simple_request(s, NG_HCI_OPCODE(NG_HCI_OGF_HC_BASEBAND,
1010 			NG_HCI_OCF_READ_VOICE_SETTINGS),
1011 			(char *) &rp, &n) == ERROR)
1012 		return (ERROR);
1013 
1014 	if (rp.status != 0x00) {
1015 		fprintf(stdout, "Status: %s [%#02x]\n",
1016 			hci_status2str(rp.status), rp.status);
1017 		return (FAILED);
1018 	}
1019 
1020 	rp.settings = le16toh(rp.settings);
1021 
1022 	input_coding      = (rp.settings & 0x0300) >> 8;
1023 	input_data_format = (rp.settings & 0x00c0) >> 6;
1024 	input_sample_size = (rp.settings & 0x0020) >> 5;
1025 
1026 	fprintf(stdout, "Voice settings: %#04x\n", rp.settings);
1027 	fprintf(stdout, "Input coding: %s [%d]\n",
1028 		hci_coding2str(input_coding), input_coding);
1029 	fprintf(stdout, "Input data format: %s [%d]\n",
1030 		hci_vdata2str(input_data_format), input_data_format);
1031 
1032 	if (input_coding == 0x00) /* Only for Linear PCM */
1033 		fprintf(stdout, "Input sample size: %d bit [%d]\n",
1034 			input_sample_size? 16 : 8, input_sample_size);
1035 
1036 	return (OK);
1037 } /* hci_read_voice_settings */
1038 
1039 /* Send Write_Voice_Settings command to the unit */
1040 static int
1041 hci_write_voice_settings(int s, int argc, char **argv)
1042 {
1043 	ng_hci_write_voice_settings_cp	cp;
1044 	ng_hci_write_voice_settings_rp	rp;
1045 	int				n;
1046 
1047 	/* parse command parameters */
1048 	switch (argc) {
1049 	case 1:
1050 		if (sscanf(argv[0], "%x", &n) != 1)
1051 			return (USAGE);
1052 
1053 		cp.settings = (uint16_t) n;
1054 		cp.settings = htole16(cp.settings);
1055 		break;
1056 
1057 	default:
1058 		return (USAGE);
1059 	}
1060 
1061 	/* send command */
1062 	n = sizeof(rp);
1063 	if (hci_request(s, NG_HCI_OPCODE(NG_HCI_OGF_HC_BASEBAND,
1064 			NG_HCI_OCF_WRITE_VOICE_SETTINGS),
1065 			(char const *) &cp, sizeof(cp),
1066 			(char *) &rp, &n) == ERROR)
1067 		return (ERROR);
1068 
1069 	if (rp.status != 0x00) {
1070 		fprintf(stdout, "Status: %s [%#02x]\n",
1071 			hci_status2str(rp.status), rp.status);
1072 		return (FAILED);
1073 	}
1074 
1075 	return (OK);
1076 } /* hci_write_voice_settings */
1077 
1078 /* Send Read_Number_Broadcast_Restransmissions */
1079 static int
1080 hci_read_number_broadcast_retransmissions(int s, int argc, char **argv)
1081 {
1082 	ng_hci_read_num_broadcast_retrans_rp	rp;
1083 	int					n;
1084 
1085 	n = sizeof(rp);
1086 	if (hci_simple_request(s, NG_HCI_OPCODE(NG_HCI_OGF_HC_BASEBAND,
1087 			NG_HCI_OCF_READ_NUM_BROADCAST_RETRANS),
1088 			(char *) &rp, &n) == ERROR)
1089 		return (ERROR);
1090 
1091 	if (rp.status != 0x00) {
1092 		fprintf(stdout, "Status: %s [%#02x]\n",
1093 			hci_status2str(rp.status), rp.status);
1094 		return (FAILED);
1095 	}
1096 
1097 	fprintf(stdout, "Number of broadcast retransmissions: %d\n",
1098 		rp.counter);
1099 
1100 	return (OK);
1101 } /* hci_read_number_broadcast_retransmissions */
1102 
1103 /* Send Write_Number_Broadcast_Restransmissions */
1104 static int
1105 hci_write_number_broadcast_retransmissions(int s, int argc, char **argv)
1106 {
1107 	ng_hci_write_num_broadcast_retrans_cp	cp;
1108 	ng_hci_write_num_broadcast_retrans_rp	rp;
1109 	int					n;
1110 
1111 	/* parse command parameters */
1112 	switch (argc) {
1113 	case 1:
1114 		if (sscanf(argv[0], "%d", &n) != 1 || n < 0 || n > 0xff)
1115 			return (USAGE);
1116 
1117 		cp.counter = (uint8_t) n;
1118 		break;
1119 
1120 	default:
1121 		return (USAGE);
1122 	}
1123 
1124 	/* send command */
1125 	n = sizeof(rp);
1126 	if (hci_request(s, NG_HCI_OPCODE(NG_HCI_OGF_HC_BASEBAND,
1127 			NG_HCI_OCF_WRITE_NUM_BROADCAST_RETRANS),
1128 			(char const *) &cp, sizeof(cp),
1129 			(char *) &rp, &n) == ERROR)
1130 		return (ERROR);
1131 
1132 	if (rp.status != 0x00) {
1133 		fprintf(stdout, "Status: %s [%#02x]\n",
1134 			hci_status2str(rp.status), rp.status);
1135 		return (FAILED);
1136 	}
1137 
1138 	return (OK);
1139 } /* hci_write_number_broadcast_retransmissions */
1140 
1141 /* Send Read_Hold_Mode_Activity command to the unit */
1142 static int
1143 hci_read_hold_mode_activity(int s, int argc, char **argv)
1144 {
1145 	ng_hci_read_hold_mode_activity_rp	rp;
1146 	int					n;
1147 	char					buffer[1024];
1148 
1149 	n = sizeof(rp);
1150 	if (hci_simple_request(s, NG_HCI_OPCODE(NG_HCI_OGF_HC_BASEBAND,
1151 			NG_HCI_OCF_READ_HOLD_MODE_ACTIVITY),
1152 			(char *) &rp, &n) == ERROR)
1153 		return (ERROR);
1154 
1155 	if (rp.status != 0x00) {
1156 		fprintf(stdout, "Status: %s [%#02x]\n",
1157 			hci_status2str(rp.status), rp.status);
1158 		return (FAILED);
1159 	}
1160 
1161 	fprintf(stdout, "Hold Mode Activities: %#02x\n", rp.hold_mode_activity);
1162 	if (rp.hold_mode_activity == 0)
1163 		fprintf(stdout, "Maintain current Power State");
1164 	else
1165 		fprintf(stdout, "%s", hci_hmode2str(rp.hold_mode_activity,
1166 				buffer, sizeof(buffer)));
1167 
1168 	fprintf(stdout, "\n");
1169 
1170 	return (OK);
1171 } /* hci_read_hold_mode_activity */
1172 
1173 /* Send Write_Hold_Mode_Activity command to the unit */
1174 static int
1175 hci_write_hold_mode_activity(int s, int argc, char **argv)
1176 {
1177 	ng_hci_write_hold_mode_activity_cp	cp;
1178 	ng_hci_write_hold_mode_activity_rp	rp;
1179 	int					n;
1180 
1181 	/* parse command parameters */
1182 	switch (argc) {
1183 	case 1:
1184 		if (sscanf(argv[0], "%d", &n) != 1 || n < 0 || n > 4)
1185 			return (USAGE);
1186 
1187 		cp.hold_mode_activity = (uint8_t) n;
1188 		break;
1189 
1190 	default:
1191 		return (USAGE);
1192 	}
1193 
1194 	/* send command */
1195 	n = sizeof(rp);
1196 	if (hci_request(s, NG_HCI_OPCODE(NG_HCI_OGF_HC_BASEBAND,
1197 			NG_HCI_OCF_WRITE_HOLD_MODE_ACTIVITY),
1198 			(char const *) &cp, sizeof(cp),
1199 			(char *) &rp, &n) == ERROR)
1200 		return (ERROR);
1201 
1202 	if (rp.status != 0x00) {
1203 		fprintf(stdout, "Status: %s [%#02x]\n",
1204 			hci_status2str(rp.status), rp.status);
1205 		return (FAILED);
1206 	}
1207 
1208 	return (OK);
1209 } /* hci_write_hold_mode_activity */
1210 
1211 /* Send Read_SCO_Flow_Control_Enable command to the unit */
1212 static int
1213 hci_read_sco_flow_control_enable(int s, int argc, char **argv)
1214 {
1215 	ng_hci_read_sco_flow_control_rp	rp;
1216 	int				n;
1217 
1218 	n = sizeof(rp);
1219 	if (hci_simple_request(s, NG_HCI_OPCODE(NG_HCI_OGF_HC_BASEBAND,
1220 			NG_HCI_OCF_READ_SCO_FLOW_CONTROL),
1221 			(char *) &rp, &n) == ERROR)
1222 		return (ERROR);
1223 
1224 	if (rp.status != 0x00) {
1225 		fprintf(stdout, "Status: %s [%#02x]\n",
1226 			hci_status2str(rp.status), rp.status);
1227 		return (FAILED);
1228 	}
1229 
1230 	fprintf(stdout, "SCO flow control %s [%d]\n",
1231 		rp.flow_control? "enabled" : "disabled", rp.flow_control);
1232 
1233 	return (OK);
1234 } /* hci_read_sco_flow_control_enable */
1235 
1236 /* Send Write_SCO_Flow_Control_Enable command to the unit */
1237 static int
1238 hci_write_sco_flow_control_enable(int s, int argc, char **argv)
1239 {
1240 	ng_hci_write_sco_flow_control_cp	cp;
1241 	ng_hci_write_sco_flow_control_rp	rp;
1242 	int					n;
1243 
1244 	/* parse command parameters */
1245 	switch (argc) {
1246 	case 1:
1247 		if (sscanf(argv[0], "%d", &n) != 1 || n < 0 || n > 1)
1248 			return (USAGE);
1249 
1250 		cp.flow_control = (uint8_t) n;
1251 		break;
1252 
1253 	default:
1254 		return (USAGE);
1255 	}
1256 
1257 	/* send command */
1258 	n = sizeof(rp);
1259 	if (hci_request(s, NG_HCI_OPCODE(NG_HCI_OGF_HC_BASEBAND,
1260 			NG_HCI_OCF_WRITE_SCO_FLOW_CONTROL),
1261 			(char const *) &cp, sizeof(cp),
1262 			(char *) &rp, &n) == ERROR)
1263 		return (ERROR);
1264 
1265 	if (rp.status != 0x00) {
1266 		fprintf(stdout, "Status: %s [%#02x]\n",
1267 			hci_status2str(rp.status), rp.status);
1268 		return (FAILED);
1269 	}
1270 
1271 	return (OK);
1272 } /* hci_write_sco_flow_control_enable */
1273 
1274 /* Send Read_Link_Supervision_Timeout command to the unit */
1275 static int
1276 hci_read_link_supervision_timeout(int s, int argc, char **argv)
1277 {
1278 	ng_hci_read_link_supervision_timo_cp	cp;
1279 	ng_hci_read_link_supervision_timo_rp	rp;
1280 	int					n;
1281 
1282 	switch (argc) {
1283 	case 1:
1284 		/* connection handle */
1285 		if (sscanf(argv[0], "%d", &n) != 1 || n <= 0 || n > 0x0eff)
1286 			return (USAGE);
1287 
1288 		cp.con_handle = (uint16_t) (n & 0x0fff);
1289 		cp.con_handle = htole16(cp.con_handle);
1290 		break;
1291 
1292 	default:
1293 		return (USAGE);
1294 	}
1295 
1296 	/* send command */
1297 	n = sizeof(rp);
1298 	if (hci_request(s, NG_HCI_OPCODE(NG_HCI_OGF_HC_BASEBAND,
1299 			NG_HCI_OCF_READ_LINK_SUPERVISION_TIMO),
1300 			(char const *) &cp, sizeof(cp),
1301 			(char *) &rp, &n) == ERROR)
1302 		return (ERROR);
1303 
1304 	if (rp.status != 0x00) {
1305 		fprintf(stdout, "Status: %s [%#02x]\n",
1306 			hci_status2str(rp.status), rp.status);
1307 		return (FAILED);
1308 	}
1309 
1310 	rp.timeout = le16toh(rp.timeout);
1311 
1312 	fprintf(stdout, "Connection handle: %d\n", le16toh(rp.con_handle));
1313 	fprintf(stdout, "Link supervision timeout: %.2f msec [%d slots]\n",
1314 		rp.timeout * 0.625, rp.timeout);
1315 
1316 	return (OK);
1317 } /* hci_read_link_supervision_timeout */
1318 
1319 /* Send Write_Link_Supervision_Timeout command to the unit */
1320 static int
1321 hci_write_link_supervision_timeout(int s, int argc, char **argv)
1322 {
1323 	ng_hci_write_link_supervision_timo_cp	cp;
1324 	ng_hci_write_link_supervision_timo_rp	rp;
1325 	int					n;
1326 
1327 	switch (argc) {
1328 	case 2:
1329 		/* connection handle */
1330 		if (sscanf(argv[0], "%d", &n) != 1 || n <= 0 || n > 0x0eff)
1331 			return (USAGE);
1332 
1333 		cp.con_handle = (uint16_t) (n & 0x0fff);
1334 		cp.con_handle = htole16(cp.con_handle);
1335 
1336 		/* link supervision timeout */
1337 		if (sscanf(argv[1], "%d", &n) != 1 || n < 0 || n > 0xffff)
1338 			return (USAGE);
1339 
1340 		cp.timeout = (uint16_t) (n & 0x0fff);
1341 		cp.timeout = htole16(cp.timeout);
1342 		break;
1343 
1344 	default:
1345 		return (USAGE);
1346 	}
1347 
1348 	/* send command */
1349 	n = sizeof(rp);
1350 	if (hci_request(s, NG_HCI_OPCODE(NG_HCI_OGF_HC_BASEBAND,
1351 			NG_HCI_OCF_WRITE_LINK_SUPERVISION_TIMO),
1352 			(char const *) &cp, sizeof(cp),
1353 			(char *) &rp, &n) == ERROR)
1354 		return (ERROR);
1355 
1356 	if (rp.status != 0x00) {
1357 		fprintf(stdout, "Status: %s [%#02x]\n",
1358 			hci_status2str(rp.status), rp.status);
1359 		return (FAILED);
1360 	}
1361 
1362 	return (OK);
1363 } /* hci_write_link_supervision_timeout */
1364 
1365 /* Send Read_Page_Scan_Period_Mode command to the unit */
1366 static int
1367 hci_read_page_scan_period_mode(int s, int argc, char **argv)
1368 {
1369 	ng_hci_read_page_scan_period_rp	rp;
1370 	int				n;
1371 
1372 	n = sizeof(rp);
1373 	if (hci_simple_request(s, NG_HCI_OPCODE(NG_HCI_OGF_HC_BASEBAND,
1374 			NG_HCI_OCF_READ_PAGE_SCAN_PERIOD),
1375 			(char *) &rp, &n) == ERROR)
1376 		return (ERROR);
1377 
1378 	if (rp.status != 0x00) {
1379 		fprintf(stdout, "Status: %s [%#02x]\n",
1380 			hci_status2str(rp.status), rp.status);
1381 		return (FAILED);
1382 	}
1383 
1384 	fprintf(stdout, "Page scan period mode: %#02x\n",
1385 		rp.page_scan_period_mode);
1386 
1387 	return (OK);
1388 } /* hci_read_page_scan_period_mode */
1389 
1390 /* Send Write_Page_Scan_Period_Mode command to the unit */
1391 static int
1392 hci_write_page_scan_period_mode(int s, int argc, char **argv)
1393 {
1394 	ng_hci_write_page_scan_period_cp	cp;
1395 	ng_hci_write_page_scan_period_rp	rp;
1396 	int					n;
1397 
1398 	/* parse command arguments */
1399 	switch (argc) {
1400 	case 1:
1401 		if (sscanf(argv[0], "%d", &n) != 1 || n < 0 || n > 2)
1402 			return (USAGE);
1403 
1404 		cp.page_scan_period_mode = (n & 0xff);
1405 		break;
1406 
1407 	default:
1408 		return (USAGE);
1409 	}
1410 
1411 	/* send command */
1412 	n = sizeof(rp);
1413 	if (hci_request(s, NG_HCI_OPCODE(NG_HCI_OGF_HC_BASEBAND,
1414 			NG_HCI_OCF_WRITE_PAGE_SCAN_PERIOD),
1415 			(char const *) &cp, sizeof(cp),
1416 			(char *) &rp, &n) == ERROR)
1417 		return (ERROR);
1418 
1419 	if (rp.status != 0x00) {
1420 		fprintf(stdout, "Status: %s [%#02x]\n",
1421 			hci_status2str(rp.status), rp.status);
1422 		return (FAILED);
1423 	}
1424 
1425 	return (OK);
1426 } /* hci_write_page_scan_period_mode */
1427 
1428 /* Send Read_Page_Scan_Mode command to the unit */
1429 static int
1430 hci_read_page_scan_mode(int s, int argc, char **argv)
1431 {
1432 	ng_hci_read_page_scan_rp	rp;
1433 	int				n;
1434 
1435 	n = sizeof(rp);
1436 	if (hci_simple_request(s, NG_HCI_OPCODE(NG_HCI_OGF_HC_BASEBAND,
1437 			NG_HCI_OCF_READ_PAGE_SCAN),
1438 			(char *) &rp, &n) == ERROR)
1439 		return (ERROR);
1440 
1441 	if (rp.status != 0x00) {
1442 		fprintf(stdout, "Status: %s [%#02x]\n",
1443 			hci_status2str(rp.status), rp.status);
1444 		return (FAILED);
1445 	}
1446 
1447 	fprintf(stdout, "Page scan mode: %#02x\n", rp.page_scan_mode);
1448 
1449 	return (OK);
1450 } /* hci_read_page_scan_mode */
1451 
1452 /* Send Write_Page_Scan_Mode command to the unit */
1453 static int
1454 hci_write_page_scan_mode(int s, int argc, char **argv)
1455 {
1456 	ng_hci_write_page_scan_cp	cp;
1457 	ng_hci_write_page_scan_rp	rp;
1458 	int				n;
1459 
1460 	/* parse command arguments */
1461 	switch (argc) {
1462 	case 1:
1463 		if (sscanf(argv[0], "%d", &n) != 1 || n < 0 || n > 3)
1464 			return (USAGE);
1465 
1466 		cp.page_scan_mode = (n & 0xff);
1467 		break;
1468 
1469 	default:
1470 		return (USAGE);
1471 	}
1472 
1473 	/* send command */
1474 	n = sizeof(rp);
1475 	if (hci_request(s, NG_HCI_OPCODE(NG_HCI_OGF_HC_BASEBAND,
1476 			NG_HCI_OCF_WRITE_PAGE_SCAN),
1477 			(char const *) &cp, sizeof(cp),
1478 			(char *) &rp, &n) == ERROR)
1479 		return (ERROR);
1480 
1481 	if (rp.status != 0x00) {
1482 		fprintf(stdout, "Status: %s [%#02x]\n",
1483 			hci_status2str(rp.status), rp.status);
1484 		return (FAILED);
1485 	}
1486 
1487 	return (OK);
1488 } /* hci_write_page_scan_mode */
1489 
1490 struct hci_command	host_controller_baseband_commands[] = {
1491 {
1492 "reset",
1493 "\nThe Reset command will reset the Host Controller and the Link Manager.\n" \
1494 "After the reset is completed, the current operational state will be lost,\n" \
1495 "the Bluetooth unit will enter standby mode and the Host Controller will\n" \
1496 "automatically revert to the default values for the parameters for which\n" \
1497 "default values are defined in the specification.",
1498 &hci_reset
1499 },
1500 {
1501 "read_pin_type",
1502 "\nThe Read_PIN_Type command is used for the Host to read whether the Link\n" \
1503 "Manager assumes that the Host supports variable PIN codes only a fixed PIN\n" \
1504 "code.",
1505 &hci_read_pin_type
1506 },
1507 {
1508 "write_pin_type <pin_type>",
1509 "\nThe Write_PIN_Type command is used for the Host to write to the Host\n" \
1510 "Controller whether the Host supports variable PIN codes or only a fixed PIN\n"\
1511 "code.\n\n" \
1512 "\t<pin_type> - dd; 0 - Variable; 1 - Fixed",
1513 &hci_write_pin_type
1514 },
1515 {
1516 "read_stored_link_key [<BD_ADDR>]",
1517 "\nThe Read_Stored_Link_Key command provides the ability to read one or\n" \
1518 "more link keys stored in the Bluetooth Host Controller. The Bluetooth Host\n" \
1519 "Controller can store a limited number of link keys for other Bluetooth\n" \
1520 "devices.\n\n" \
1521 "\t<BD_ADDR> - xx:xx:xx:xx:xx:xx BD_ADDR or name",
1522 &hci_read_stored_link_key
1523 },
1524 {
1525 "write_stored_link_key <BD_ADDR> <key>",
1526 "\nThe Write_Stored_Link_Key command provides the ability to write one\n" \
1527 "or more link keys to be stored in the Bluetooth Host Controller. The\n" \
1528 "Bluetooth Host Controller can store a limited number of link keys for other\n"\
1529 "Bluetooth devices. If no additional space is available in the Bluetooth\n"\
1530 "Host Controller then no additional link keys will be stored.\n\n" \
1531 "\t<BD_ADDR> - xx:xx:xx:xx:xx:xx BD_ADDR or name\n" \
1532 "\t<key>     - xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx up to 16 bytes link key",
1533 &hci_write_stored_link_key
1534 },
1535 {
1536 "delete_stored_link_key [<BD_ADDR>]",
1537 "\nThe Delete_Stored_Link_Key command provides the ability to remove one\n" \
1538 "or more of the link keys stored in the Bluetooth Host Controller. The\n" \
1539 "Bluetooth Host Controller can store a limited number of link keys for other\n"\
1540 "Bluetooth devices.\n\n" \
1541 "\t<BD_ADDR> - xx:xx:xx:xx:xx:xx BD_ADDR or name",
1542 &hci_delete_stored_link_key
1543 },
1544 {
1545 "change_local_name <name>",
1546 "\nThe Change_Local_Name command provides the ability to modify the user\n" \
1547 "friendly name for the Bluetooth unit.\n\n" \
1548 "\t<name> - string",
1549 &hci_change_local_name
1550 },
1551 {
1552 "read_local_name",
1553 "\nThe Read_Local_Name command provides the ability to read the\n" \
1554 "stored user-friendly name for the Bluetooth unit.",
1555 &hci_read_local_name
1556 },
1557 {
1558 "read_connection_accept_timeout",
1559 "\nThis command will read the value for the Connection_Accept_Timeout\n" \
1560 "configuration parameter. The Connection_Accept_Timeout configuration\n" \
1561 "parameter allows the Bluetooth hardware to automatically deny a\n" \
1562 "connection request after a specified time period has occurred and\n" \
1563 "the new connection is not accepted. Connection Accept Timeout\n" \
1564 "measured in Number of Baseband slots.",
1565 &hci_read_connection_accept_timeout
1566 },
1567 {
1568 "write_connection_accept_timeout <timeout>",
1569 "\nThis command will write the value for the Connection_Accept_Timeout\n" \
1570 "configuration parameter.\n\n" \
1571 "\t<timeout> - dddd; measured in number of baseband slots.",
1572 &hci_write_connection_accept_timeout
1573 },
1574 {
1575 "read_page_timeout",
1576 "\nThis command will read the value for the Page_Timeout configuration\n" \
1577 "parameter. The Page_Timeout configuration parameter defines the\n" \
1578 "maximum time the local Link Manager will wait for a baseband page\n" \
1579 "response from the remote unit at a locally initiated connection\n" \
1580 "attempt. Page Timeout measured in Number of Baseband slots.",
1581 &hci_read_page_timeout
1582 },
1583 {
1584 "write_page_timeout <timeout>",
1585 "\nThis command will write the value for the Page_Timeout configuration\n" \
1586 "parameter.\n\n" \
1587 "\t<timeout> - dddd; measured in number of baseband slots.",
1588 &hci_write_page_timeout
1589 },
1590 {
1591 "read_scan_enable",
1592 "\nThis command will read the value for the Scan_Enable parameter. The\n" \
1593 "Scan_Enable parameter controls whether or not the Bluetooth uint\n" \
1594 "will periodically scan for page attempts and/or inquiry requests\n" \
1595 "from other Bluetooth unit.\n\n" \
1596 "\t0x00 - No Scans enabled.\n" \
1597 "\t0x01 - Inquiry Scan enabled. Page Scan disabled.\n" \
1598 "\t0x02 - Inquiry Scan disabled. Page Scan enabled.\n" \
1599 "\t0x03 - Inquiry Scan enabled. Page Scan enabled.",
1600 &hci_read_scan_enable
1601 },
1602 {
1603 "write_scan_enable <scan_enable>",
1604 "\nThis command will write the value for the Scan_Enable parameter.\n" \
1605 "The Scan_Enable parameter controls whether or not the Bluetooth\n" \
1606 "unit will periodically scan for page attempts and/or inquiry\n" \
1607 "requests from other Bluetooth unit.\n\n" \
1608 "\t<scan_enable> - dd;\n" \
1609 "\t0 - No Scans enabled.\n" \
1610 "\t1 - Inquiry Scan enabled. Page Scan disabled.\n" \
1611 "\t2 - Inquiry Scan disabled. Page Scan enabled.\n" \
1612 "\t3 - Inquiry Scan enabled. Page Scan enabled.",
1613 &hci_write_scan_enable
1614 },
1615 {
1616 "read_page_scan_activity",
1617 "\nThis command will read the value for Page_Scan_Activity configuration\n" \
1618 "parameters. The Page_Scan_Interval configuration parameter defines the\n" \
1619 "amount of time between consecutive page scans. This time interval is \n" \
1620 "defined from when the Host Controller started its last page scan until\n" \
1621 "it begins the next page scan. The Page_Scan_Window configuration parameter\n" \
1622 "defines the amount of time for the duration of the page scan. The\n" \
1623 "Page_Scan_Window can only be less than or equal to the Page_Scan_Interval.",
1624 &hci_read_page_scan_activity
1625 },
1626 {
1627 "write_page_scan_activity interval(dddd) window(dddd)",
1628 "\nThis command will write the value for Page_Scan_Activity configuration\n" \
1629 "parameter. The Page_Scan_Interval configuration parameter defines the\n" \
1630 "amount of time between consecutive page scans. This is defined as the time\n" \
1631 "interval from when the Host Controller started its last page scan until it\n" \
1632 "begins the next page scan. The Page_Scan_Window configuration parameter\n" \
1633 "defines the amount of time for the duration of the page scan. \n" \
1634 "The Page_Scan_Window can only be less than or equal to the Page_Scan_Interval.\n\n" \
1635 "\t<interval> - Range: 0x0012 -- 0x100, Time = N * 0.625 msec\n" \
1636 "\t<window>   - Range: 0x0012 -- 0x100, Time = N * 0.625 msec",
1637 &hci_write_page_scan_activity
1638 },
1639 {
1640 "read_inquiry_scan_activity",
1641 "\nThis command will read the value for Inquiry_Scan_Activity configuration\n" \
1642 "parameter. The Inquiry_Scan_Interval configuration parameter defines the\n" \
1643 "amount of time between consecutive inquiry scans. This is defined as the\n" \
1644 "time interval from when the Host Controller started its last inquiry scan\n" \
1645 "until it begins the next inquiry scan.",
1646 &hci_read_inquiry_scan_activity
1647 },
1648 {
1649 "write_inquiry_scan_activity interval(dddd) window(dddd)",
1650 "\nThis command will write the value for Inquiry_Scan_Activity configuration\n"\
1651 "parameter. The Inquiry_Scan_Interval configuration parameter defines the\n" \
1652 "amount of time between consecutive inquiry scans. This is defined as the\n" \
1653 "time interval from when the Host Controller started its last inquiry scan\n" \
1654 "until it begins the next inquiry scan. The Inquiry_Scan_Window configuration\n" \
1655 "parameter defines the amount of time for the duration of the inquiry scan.\n" \
1656 "The Inquiry_Scan_Window can only be less than or equal to the Inquiry_Scan_Interval.\n\n" \
1657 "\t<interval> - Range: 0x0012 -- 0x100, Time = N * 0.625 msec\n" \
1658 "\t<window>   - Range: 0x0012 -- 0x100, Time = N * 0.625 msec",
1659 &hci_write_inquiry_scan_activity
1660 },
1661 {
1662 "read_authentication_enable",
1663 "\nThis command will read the value for the Authentication_Enable parameter.\n"\
1664 "The Authentication_Enable parameter controls if the local unit requires\n"\
1665 "to authenticate the remote unit at connection setup (between the\n" \
1666 "Create_Connection command or acceptance of an incoming ACL connection\n"\
1667 "and the corresponding Connection Complete event). At connection setup, only\n"\
1668 "the unit(s) with the Authentication_Enable parameter enabled will try to\n"\
1669 "authenticate the other unit.",
1670 &hci_read_authentication_enable
1671 },
1672 {
1673 "write_authentication_enable enable(0|1)",
1674 "\nThis command will write the value for the Authentication_Enable parameter.\n"\
1675 "The Authentication_Enable parameter controls if the local unit requires to\n"\
1676 "authenticate the remote unit at connection setup (between the\n" \
1677 "Create_Connection command or acceptance of an incoming ACL connection\n" \
1678 "and the corresponding Connection Complete event). At connection setup, only\n"\
1679 "the unit(s) with the Authentication_Enable parameter enabled will try to\n"\
1680 "authenticate the other unit.",
1681 &hci_write_authentication_enable
1682 },
1683 {
1684 "read_encryption_mode",
1685 "\nThis command will read the value for the Encryption_Mode parameter. The\n" \
1686 "Encryption_Mode parameter controls if the local unit requires encryption\n" \
1687 "to the remote unit at connection setup (between the Create_Connection\n" \
1688 "command or acceptance of an incoming ACL connection and the corresponding\n" \
1689 "Connection Complete event). At connection setup, only the unit(s) with\n" \
1690 "the Authentication_Enable parameter enabled and Encryption_Mode parameter\n" \
1691 "enabled will try to encrypt the connection to the other unit.\n\n" \
1692 "\t<encryption_mode>:\n" \
1693 "\t0x00 - Encryption disabled.\n" \
1694 "\t0x01 - Encryption only for point-to-point packets.\n" \
1695 "\t0x02 - Encryption for both point-to-point and broadcast packets.",
1696 &hci_read_encryption_mode
1697 },
1698 {
1699 "write_encryption_mode mode(0|1|2)",
1700 "\tThis command will write the value for the Encryption_Mode parameter.\n" \
1701 "The Encryption_Mode parameter controls if the local unit requires\n" \
1702 "encryption to the remote unit at connection setup (between the\n" \
1703 "Create_Connection command or acceptance of an incoming ACL connection\n" \
1704 "and the corresponding Connection Complete event). At connection setup,\n" \
1705 "only the unit(s) with the Authentication_Enable parameter enabled and\n" \
1706 "Encryption_Mode parameter enabled will try to encrypt the connection to\n" \
1707 "the other unit.\n\n" \
1708 "\t<encryption_mode> (dd)\n" \
1709 "\t0 - Encryption disabled.\n" \
1710 "\t1 - Encryption only for point-to-point packets.\n" \
1711 "\t2 - Encryption for both point-to-point and broadcast packets.",
1712 &hci_write_encryption_mode
1713 },
1714 {
1715 "read_class_of_device",
1716 "\nThis command will read the value for the Class_of_Device parameter.\n" \
1717 "The Class_of_Device parameter is used to indicate the capabilities of\n" \
1718 "the local unit to other units.",
1719 &hci_read_class_of_device
1720 },
1721 {
1722 "write_class_of_device class(xx:xx:xx)",
1723 "\nThis command will write the value for the Class_of_Device parameter.\n" \
1724 "The Class_of_Device parameter is used to indicate the capabilities of \n" \
1725 "the local unit to other units.\n\n" \
1726 "\t<class> (xx:xx:xx) - class of device",
1727 &hci_write_class_of_device
1728 },
1729 {
1730 "read_voice_settings",
1731 "\nThis command will read the values for the Voice_Setting parameter.\n" \
1732 "The Voice_Setting parameter controls all the various settings for voice\n" \
1733 "connections. These settings apply to all voice connections, and cannot be\n" \
1734 "set for individual voice connections. The Voice_Setting parameter controls\n" \
1735 "the configuration for voice connections: Input Coding, Air coding format,\n" \
1736 "input data format, Input sample size, and linear PCM parameter.",
1737 &hci_read_voice_settings
1738 },
1739 {
1740 "write_voice_settings settings(xxxx)",
1741 "\nThis command will write the values for the Voice_Setting parameter.\n" \
1742 "The Voice_Setting parameter controls all the various settings for voice\n" \
1743 "connections. These settings apply to all voice connections, and cannot be\n" \
1744 "set for individual voice connections. The Voice_Setting parameter controls\n" \
1745 "the configuration for voice connections: Input Coding, Air coding format,\n" \
1746 "input data format, Input sample size, and linear PCM parameter.\n\n" \
1747 "\t<voice_settings> (xxxx) - voice settings",
1748 &hci_write_voice_settings
1749 },
1750 {
1751 "read_number_broadcast_retransmissions",
1752 "\nThis command will read the unit's parameter value for the Number of\n" \
1753 "Broadcast Retransmissions. Broadcast packets are not acknowledged and are\n" \
1754 "unreliable.",
1755 &hci_read_number_broadcast_retransmissions
1756 },
1757 {
1758 "write_number_broadcast_retransmissions count(dd)",
1759 "\nThis command will write the unit's parameter value for the Number of\n" \
1760 "Broadcast Retransmissions. Broadcast packets are not acknowledged and are\n" \
1761 "unreliable.\n\n" \
1762 "\t<count> (dd) - number of broadcast retransimissions",
1763 &hci_write_number_broadcast_retransmissions
1764 },
1765 {
1766 "read_hold_mode_activity",
1767 "\nThis command will read the value for the Hold_Mode_Activity parameter.\n" \
1768 "The Hold_Mode_Activity value is used to determine what activities should\n" \
1769 "be suspended when the unit is in hold mode.",
1770 &hci_read_hold_mode_activity
1771 },
1772 {
1773 "write_hold_mode_activity settings(0|1|2|4)",
1774 "\nThis command will write the value for the Hold_Mode_Activity parameter.\n" \
1775 "The Hold_Mode_Activity value is used to determine what activities should\n" \
1776 "be suspended when the unit is in hold mode.\n\n" \
1777 "\t<settings> (dd) - bit mask:\n" \
1778 "\t0 - Maintain current Power State. Default\n" \
1779 "\t1 - Suspend Page Scan.\n" \
1780 "\t2 - Suspend Inquiry Scan.\n" \
1781 "\t4 - Suspend Periodic Inquiries.",
1782 &hci_write_hold_mode_activity
1783 },
1784 {
1785 "read_sco_flow_control_enable",
1786 "\nThe Read_SCO_Flow_Control_Enable command provides the ability to read\n" \
1787 "the SCO_Flow_Control_Enable setting. By using this setting, the Host can\n" \
1788 "decide if the Host Controller will send Number Of Completed Packets events\n" \
1789 "for SCO Connection Handles. This setting allows the Host to enable and\n" \
1790 "disable SCO flow control.",
1791 &hci_read_sco_flow_control_enable
1792 },
1793 {
1794 "write_sco_flow_control_enable enable(0|1)",
1795 "\nThe Write_SCO_Flow_Control_Enable command provides the ability to write\n" \
1796 "the SCO_Flow_Control_Enable setting. By using this setting, the Host can\n" \
1797 "decide if the Host Controller will send Number Of Completed Packets events\n" \
1798 "for SCO Connection Handles. This setting allows the Host to enable and\n" \
1799 "disable SCO flow control. The SCO_Flow_Control_Enable setting can only be\n" \
1800 "changed if no connections exist.",
1801 &hci_write_sco_flow_control_enable
1802 },
1803 {
1804 "read_link_supervision_timeout <connection_handle>",
1805 "\nThis command will read the value for the Link_Supervision_Timeout\n" \
1806 "parameter for the device. The Link_Supervision_Timeout parameter is used\n" \
1807 "by the master or slave Bluetooth device to monitor link loss. If, for any\n" \
1808 "reason, no Baseband packets are received from that Connection Handle for a\n" \
1809 "duration longer than the Link_Supervision_Timeout, the connection is\n"
1810 "disconnected.\n\n" \
1811 "\t<connection_handle> - dddd; connection handle\n",
1812 &hci_read_link_supervision_timeout
1813 },
1814 {
1815 "write_link_supervision_timeout <connection_handle> <timeout>",
1816 "\nThis command will write the value for the Link_Supervision_Timeout\n" \
1817 "parameter for the device. The Link_Supervision_Timeout parameter is used\n" \
1818 "by the master or slave Bluetooth device to monitor link loss. If, for any\n" \
1819 "reason, no Baseband packets are received from that connection handle for a\n" \
1820 "duration longer than the Link_Supervision_Timeout, the connection is\n" \
1821 "disconnected.\n\n" \
1822 "\t<connection_handle> - dddd; connection handle\n" \
1823 "\t<timeout>           - dddd; timeout measured in number of baseband slots\n",
1824 &hci_write_link_supervision_timeout
1825 },
1826 {
1827 "read_page_scan_period_mode",
1828 "\nThis command is used to read the mandatory Page_Scan_Period_Mode of the\n" \
1829 "local Bluetooth device. Every time an inquiry response message is sent, the\n"\
1830 "Bluetooth device will start a timer (T_mandatory_pscan), the value of which\n"\
1831 "is dependent on the Page_Scan_Period_Mode. As long as this timer has not\n" \
1832 "expired, the Bluetooth device will use the Page_Scan_Period_Mode for all\n" \
1833 "following page scans.",
1834 &hci_read_page_scan_period_mode
1835 },
1836 {
1837 "write_page_scan_period_mode <page_scan_period_mode>",
1838 "\nThis command is used to write the mandatory Page_Scan_Period_Mode of the\n" \
1839 "local Bluetooth device. Every time an inquiry response message is sent, the\n"\
1840 "Bluetooth device will start a timer (T_mandatory_pscan), the value of which\n"\
1841 "is dependent on the Page_Scan_Period_Mode. As long as this timer has not\n" \
1842 "expired, the Bluetooth device will use the Page_Scan_Period_Mode for all\n" \
1843 "following page scans.\n\n" \
1844 "\t<page_scan_period_mode> - dd; page scan period mode:\n" \
1845 "\t0x00 - P0 (Default)\n" \
1846 "\t0x01 - P1\n" \
1847 "\t0x02 - P2",
1848 &hci_write_page_scan_period_mode
1849 },
1850 {
1851 "read_page_scan_mode",
1852 "\nThis command is used to read the default page scan mode of the local\n" \
1853 "Bluetooth device. The Page_Scan_Mode parameter indicates the page scan mode\n"\
1854 "that is used for the default page scan. Currently one mandatory page scan\n"\
1855 "mode and three optional page scan modes are defined. Following an inquiry\n" \
1856 "response, if the Baseband timer T_mandatory_pscan has not expired, the\n" \
1857 "mandatory page scan mode must be applied.",
1858 &hci_read_page_scan_mode
1859 },
1860 {
1861 "write_page_scan_mode <page_scan_mode>",
1862 "\nThis command is used to write the default page scan mode of the local\n" \
1863 "Bluetooth device. The Page_Scan_Mode parameter indicates the page scan mode\n"\
1864 "that is used for the default page scan. Currently, one mandatory page scan\n"\
1865 "mode and three optional page scan modes are defined. Following an inquiry\n"\
1866 "response, if the Baseband timer T_mandatory_pscan has not expired, the\n" \
1867 "mandatory page scan mode must be applied.\n\n" \
1868 "\t<page_scan_mode> - dd; page scan mode:\n" \
1869 "\t0x00 - Mandatory Page Scan Mode (Default)\n" \
1870 "\t0x01 - Optional Page Scan Mode I\n" \
1871 "\t0x02 - Optional Page Scan Mode II\n" \
1872 "\t0x03 - Optional Page Scan Mode III",
1873 &hci_write_page_scan_mode
1874 },
1875 { NULL, }
1876 };
1877 
1878