/* $NetBSD: btconfig.c,v 1.22 2009/10/09 12:58:28 plunky Exp $ */ /*- * Copyright (c) 2006 Itronix Inc. * All rights reserved. * * Written by Iain Hibbert for Itronix Inc. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * 3. The name of Itronix Inc. may not be used to endorse * or promote products derived from this software without specific * prior written permission. * * THIS SOFTWARE IS PROVIDED BY ITRONIX INC. ``AS IS'' AND * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL ITRONIX INC. BE LIABLE FOR ANY * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND * ON ANY THEORY OF LIABILITY, WHETHER IN * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE * POSSIBILITY OF SUCH DAMAGE. */ #include __COPYRIGHT("@(#) Copyright (c) 2006 Itronix, Inc. All rights reserved."); __RCSID("$NetBSD: btconfig.c,v 1.22 2009/10/09 12:58:28 plunky Exp $"); #include #include #include #include #include #include #include #include #include #include #include int main(int, char *[]); void badarg(const char *); void badparam(const char *); void badval(const char *, const char *); void usage(void); int set_unit(unsigned long); void config_unit(void); void print_val(const char *, const char **, int); void print_info(int); void print_stats(void); void print_class(const char *, uint8_t *); void print_class0(void); void print_voice(int); void tag(const char *); void print_features(const char *, uint8_t, uint8_t *); void print_features0(uint8_t *); void print_features1(uint8_t *); void print_result(int, struct bt_devinquiry *); void do_inquiry(void); void hci_req(uint16_t, uint8_t , void *, size_t, void *, size_t); void save_value(uint16_t, void *, size_t); void load_value(uint16_t, void *, size_t); #define MAX_STR_SIZE 0xff /* print width */ int width = 0; #define MAX_WIDTH 70 /* global variables */ int hci; struct btreq btr; /* command line flags */ int verbose = 0; /* more info */ int lflag = 0; /* list devices */ int sflag = 0; /* get/zero stats */ /* device up/down (flag) */ int opt_enable = 0; int opt_reset = 0; #define FLAGS_FMT "\20" \ "\001UP" \ "\002RUNNING" \ "\003XMIT_CMD" \ "\004XMIT_ACL" \ "\005XMIT_SCO" \ "\006INIT_BDADDR" \ "\007INIT_BUFFER_SIZE" \ "\010INIT_FEATURES" \ "\011POWER_UP_NOOP" \ "\012INIT_COMMANDS" \ "\013MASTER" \ "" /* authorisation (flag) */ int opt_auth = 0; /* encryption (flag) */ int opt_encrypt = 0; /* scan enable options (flags) */ int opt_pscan = 0; int opt_iscan = 0; /* master role option */ int opt_master = 0; /* link policy options (flags) */ int opt_switch = 0; int opt_hold = 0; int opt_sniff = 0; int opt_park = 0; /* class of device (hex value) */ int opt_class = 0; uint32_t class; /* packet type mask (hex value) */ int opt_ptype = 0; uint32_t ptype; /* unit name (string) */ int opt_name = 0; char name[MAX_STR_SIZE]; /* pin type */ int opt_pin = 0; /* Inquiry */ int opt_rssi = 0; /* inquiry_with_rssi (obsolete flag) */ int opt_imode = 0; /* inquiry mode */ int opt_inquiry = 0; #define INQUIRY_LENGTH 10 /* seconds */ #define INQUIRY_MAX_RESPONSES 10 const char *imodes[] = { "std", "rssi", "ext", NULL }; /* Voice Settings */ int opt_voice = 0; uint32_t voice; /* Page Timeout */ int opt_pto = 0; uint32_t pto; /* set SCO mtu */ int opt_scomtu; uint32_t scomtu; struct parameter { const char *name; enum { P_SET, P_CLR, P_STR, P_HEX, P_NUM, P_VAL } type; int *opt; void *val; } parameters[] = { { "up", P_SET, &opt_enable, NULL }, { "enable", P_SET, &opt_enable, NULL }, { "down", P_CLR, &opt_enable, NULL }, { "disable", P_CLR, &opt_enable, NULL }, { "name", P_STR, &opt_name, name }, { "pscan", P_SET, &opt_pscan, NULL }, { "-pscan", P_CLR, &opt_pscan, NULL }, { "iscan", P_SET, &opt_iscan, NULL }, { "-iscan", P_CLR, &opt_iscan, NULL }, { "master", P_SET, &opt_master, NULL }, { "-master", P_CLR, &opt_master, NULL }, { "switch", P_SET, &opt_switch, NULL }, { "-switch", P_CLR, &opt_switch, NULL }, { "hold", P_SET, &opt_hold, NULL }, { "-hold", P_CLR, &opt_hold, NULL }, { "sniff", P_SET, &opt_sniff, NULL }, { "-sniff", P_CLR, &opt_sniff, NULL }, { "park", P_SET, &opt_park, NULL }, { "-park", P_CLR, &opt_park, NULL }, { "auth", P_SET, &opt_auth, NULL }, { "-auth", P_CLR, &opt_auth, NULL }, { "encrypt", P_SET, &opt_encrypt, NULL }, { "-encrypt", P_CLR, &opt_encrypt, NULL }, { "ptype", P_HEX, &opt_ptype, &ptype }, { "class", P_HEX, &opt_class, &class }, { "fixed", P_SET, &opt_pin, NULL }, { "variable", P_CLR, &opt_pin, NULL }, { "inq", P_SET, &opt_inquiry, NULL }, { "inquiry", P_SET, &opt_inquiry, NULL }, { "imode", P_VAL, &opt_imode, imodes }, { "rssi", P_SET, &opt_rssi, NULL }, { "-rssi", P_CLR, &opt_rssi, NULL }, { "reset", P_SET, &opt_reset, NULL }, { "voice", P_HEX, &opt_voice, &voice }, { "pto", P_NUM, &opt_pto, &pto }, { "scomtu", P_NUM, &opt_scomtu, &scomtu }, { NULL, 0, NULL, NULL } }; int main(int ac, char *av[]) { int ch; struct parameter *p; while ((ch = getopt(ac, av, "hlsvz")) != -1) { switch(ch) { case 'l': lflag = 1; break; case 's': sflag = 1; break; case 'v': verbose++; break; case 'z': sflag = 2; break; case 'h': default: usage(); } } av += optind; ac -= optind; if (lflag && sflag) usage(); hci = socket(PF_BLUETOOTH, SOCK_RAW, BTPROTO_HCI); if (hci == -1) err(EXIT_FAILURE, "socket"); if (ac == 0) { verbose++; memset(&btr, 0, sizeof(btr)); while (set_unit(SIOCNBTINFO) != -1) { print_info(verbose); print_stats(); } tag(NULL); } else { strlcpy(btr.btr_name, *av, HCI_DEVNAME_SIZE); av++, ac--; if (set_unit(SIOCGBTINFO) < 0) err(EXIT_FAILURE, "%s", btr.btr_name); if (ac == 0) verbose += 2; while (ac > 0) { for (p = parameters ; ; p++) { if (p->name == NULL) badparam(*av); if (strcmp(*av, p->name) == 0) break; } switch(p->type) { case P_SET: *(p->opt) = 1; break; case P_CLR: *(p->opt) = -1; break; case P_STR: if (--ac < 1) badarg(p->name); strlcpy((char *)(p->val), *++av, MAX_STR_SIZE); *(p->opt) = 1; break; case P_HEX: if (--ac < 1) badarg(p->name); *(uint32_t *)(p->val) = strtoul(*++av, NULL, 16); *(p->opt) = 1; break; case P_NUM: if (--ac < 1) badarg(p->name); *(uint32_t *)(p->val) = strtoul(*++av, NULL, 10); *(p->opt) = 1; break; case P_VAL: if (--ac < 1) badarg(p->name); ++av; ch = 0; do { if (((char **)(p->val))[ch] == NULL) badval(p->name, *av); } while (strcmp(((char **)(p->val))[ch++], *av)); *(p->opt) = ch; break; } av++, ac--; } config_unit(); print_info(verbose); print_stats(); do_inquiry(); } close(hci); return EXIT_SUCCESS; } void badparam(const char *param) { fprintf(stderr, "unknown parameter '%s'\n", param); exit(EXIT_FAILURE); } void badarg(const char *param) { fprintf(stderr, "parameter '%s' needs argument\n", param); exit(EXIT_FAILURE); } void badval(const char *param, const char *value) { fprintf(stderr, "bad value '%s' for parameter '%s'\n", value, param); exit(EXIT_FAILURE); } void usage(void) { fprintf(stderr, "usage: %s [-svz] [device [parameters]]\n", getprogname()); fprintf(stderr, " %s -l\n", getprogname()); exit(EXIT_FAILURE); } /* * pretty printing feature */ void tag(const char *f) { if (f == NULL) { if (width > 0) printf("\n"); width = 0; } else { width += printf("%*s%s", (width == 0 ? (lflag ? 0 : 8) : 1), "", f); if (width > MAX_WIDTH) { printf("\n"); width = 0; } } } /* * basic HCI request wrapper with error check */ void hci_req(uint16_t opcode, uint8_t event, void *cbuf, size_t clen, void *rbuf, size_t rlen) { struct bt_devreq req; req.opcode = opcode; req.event = event; req.cparam = cbuf; req.clen = clen; req.rparam = rbuf; req.rlen = rlen; if (bt_devreq(hci, &req, 10) == -1) err(EXIT_FAILURE, "cmd (%02x|%03x)", HCI_OGF(opcode), HCI_OCF(opcode)); if (event == 0 && rlen > 0 && ((uint8_t *)rbuf)[0] != 0) errx(EXIT_FAILURE, "cmd (%02x|%03x): status 0x%02x", HCI_OGF(opcode), HCI_OCF(opcode), ((uint8_t *)rbuf)[0]); } /* * write value to device with opcode. * provide a small response buffer so that the status can be checked */ void save_value(uint16_t opcode, void *cbuf, size_t clen) { uint8_t buf[1]; hci_req(opcode, 0, cbuf, clen, buf, sizeof(buf)); } /* * read value from device with opcode. * use our own buffer and only return the value from the response packet */ void load_value(uint16_t opcode, void *rbuf, size_t rlen) { uint8_t buf[UINT8_MAX]; hci_req(opcode, 0, NULL, 0, buf, sizeof(buf)); memcpy(rbuf, buf + 1, rlen); } int set_unit(unsigned long cmd) { if (ioctl(hci, cmd, &btr) == -1) return -1; if (btr.btr_flags & BTF_UP) { struct sockaddr_bt sa; sa.bt_len = sizeof(sa); sa.bt_family = AF_BLUETOOTH; bdaddr_copy(&sa.bt_bdaddr, &btr.btr_bdaddr); if (bind(hci, (struct sockaddr *)&sa, sizeof(sa)) < 0) err(EXIT_FAILURE, "bind"); if (connect(hci, (struct sockaddr *)&sa, sizeof(sa)) < 0) err(EXIT_FAILURE, "connect"); } return 0; } /* * apply configuration parameters to unit */ void config_unit(void) { if (opt_enable) { if (opt_enable > 0) btr.btr_flags |= BTF_UP; else btr.btr_flags &= ~BTF_UP; if (ioctl(hci, SIOCSBTFLAGS, &btr) < 0) err(EXIT_FAILURE, "SIOCSBTFLAGS"); if (set_unit(SIOCGBTINFO) < 0) err(EXIT_FAILURE, "%s", btr.btr_name); } if (opt_reset) { hci_req(HCI_CMD_RESET, 0, NULL, 0, NULL, 0); btr.btr_flags |= BTF_INIT; if (ioctl(hci, SIOCSBTFLAGS, &btr) < 0) err(EXIT_FAILURE, "SIOCSBTFLAGS"); /* * although the reset command will automatically * carry out these commands, we do them manually * just so we can wait for completion. */ hci_req(HCI_CMD_READ_BDADDR, 0, NULL, 0, NULL, 0); hci_req(HCI_CMD_READ_BUFFER_SIZE, 0, NULL, 0, NULL, 0); hci_req(HCI_CMD_READ_LOCAL_FEATURES, 0, NULL, 0, NULL, 0); if (set_unit(SIOCGBTINFO) < 0) err(EXIT_FAILURE, "%s", btr.btr_name); } if (opt_master) { if (opt_master > 0) btr.btr_flags |= BTF_MASTER; else btr.btr_flags &= ~BTF_MASTER; if (ioctl(hci, SIOCSBTFLAGS, &btr) < 0) err(EXIT_FAILURE, "SIOCSBTFLAGS"); } if (opt_switch || opt_hold || opt_sniff || opt_park) { uint16_t val = btr.btr_link_policy; if (opt_switch > 0) val |= HCI_LINK_POLICY_ENABLE_ROLE_SWITCH; if (opt_switch < 0) val &= ~HCI_LINK_POLICY_ENABLE_ROLE_SWITCH; if (opt_hold > 0) val |= HCI_LINK_POLICY_ENABLE_HOLD_MODE; if (opt_hold < 0) val &= ~HCI_LINK_POLICY_ENABLE_HOLD_MODE; if (opt_sniff > 0) val |= HCI_LINK_POLICY_ENABLE_SNIFF_MODE; if (opt_sniff < 0) val &= ~HCI_LINK_POLICY_ENABLE_SNIFF_MODE; if (opt_park > 0) val |= HCI_LINK_POLICY_ENABLE_PARK_MODE; if (opt_park < 0) val &= ~HCI_LINK_POLICY_ENABLE_PARK_MODE; btr.btr_link_policy = val; if (ioctl(hci, SIOCSBTPOLICY, &btr) < 0) err(EXIT_FAILURE, "SIOCSBTPOLICY"); } if (opt_ptype) { btr.btr_packet_type = ptype; if (ioctl(hci, SIOCSBTPTYPE, &btr) < 0) err(EXIT_FAILURE, "SIOCSBTPTYPE"); } if (opt_pscan || opt_iscan) { uint8_t val; load_value(HCI_CMD_READ_SCAN_ENABLE, &val, sizeof(val)); if (opt_pscan > 0) val |= HCI_PAGE_SCAN_ENABLE; if (opt_pscan < 0) val &= ~HCI_PAGE_SCAN_ENABLE; if (opt_iscan > 0) val |= HCI_INQUIRY_SCAN_ENABLE; if (opt_iscan < 0) val &= ~HCI_INQUIRY_SCAN_ENABLE; save_value(HCI_CMD_WRITE_SCAN_ENABLE, &val, sizeof(val)); } if (opt_auth) { uint8_t val = (opt_auth > 0 ? 1 : 0); save_value(HCI_CMD_WRITE_AUTH_ENABLE, &val, sizeof(val)); } if (opt_encrypt) { uint8_t val = (opt_encrypt > 0 ? 1 : 0); save_value(HCI_CMD_WRITE_ENCRYPTION_MODE, &val, sizeof(val)); } if (opt_name) save_value(HCI_CMD_WRITE_LOCAL_NAME, name, HCI_UNIT_NAME_SIZE); if (opt_class) { uint8_t val[HCI_CLASS_SIZE]; val[0] = (class >> 0) & 0xff; val[1] = (class >> 8) & 0xff; val[2] = (class >> 16) & 0xff; save_value(HCI_CMD_WRITE_UNIT_CLASS, val, HCI_CLASS_SIZE); } if (opt_pin) { uint8_t val; if (opt_pin > 0) val = 1; else val = 0; save_value(HCI_CMD_WRITE_PIN_TYPE, &val, sizeof(val)); } if (opt_voice) { uint16_t val; val = htole16(voice & 0x03ff); save_value(HCI_CMD_WRITE_VOICE_SETTING, &val, sizeof(val)); } if (opt_pto) { uint16_t val; val = htole16(pto * 8 / 5); save_value(HCI_CMD_WRITE_PAGE_TIMEOUT, &val, sizeof(val)); } if (opt_scomtu) { if (scomtu > 0xff) { warnx("Invalid SCO mtu %d", scomtu); } else { btr.btr_sco_mtu = scomtu; if (ioctl(hci, SIOCSBTSCOMTU, &btr) < 0) warn("SIOCSBTSCOMTU"); } } if (opt_imode | opt_rssi) { uint8_t val = (opt_rssi > 0 ? 1 : 0); if (opt_imode) val = opt_imode - 1; save_value(HCI_CMD_WRITE_INQUIRY_MODE, &val, sizeof(val)); } } /* * print value from NULL terminated array given index */ void print_val(const char *hdr, const char **argv, int idx) { int i = 0; while (i < idx && *argv != NULL) i++, argv++; printf("\t%s: %s\n", hdr, *argv == NULL ? "unknown" : *argv); } /* * Print info for Bluetooth Device with varying verbosity levels */ void print_info(int level) { uint8_t version, val, buf[MAX_STR_SIZE]; if (lflag) { tag(btr.btr_name); return; } if (level-- < 1) return; snprintb((char *)buf, MAX_STR_SIZE, FLAGS_FMT, btr.btr_flags); printf("%s: bdaddr %s flags %s\n", btr.btr_name, bt_ntoa(&btr.btr_bdaddr, NULL), buf); if (level-- < 1) return; printf("\tnum_cmd = %d\n" "\tnum_acl = %d, acl_mtu = %d\n" "\tnum_sco = %d, sco_mtu = %d\n", btr.btr_num_cmd, btr.btr_num_acl, btr.btr_acl_mtu, btr.btr_num_sco, btr.btr_sco_mtu); if (level-- < 1 || (btr.btr_flags & BTF_UP) == 0) return; load_value(HCI_CMD_READ_LOCAL_VER, &version, sizeof(version)); printf("\tHCI version: "); switch(version) { case HCI_SPEC_V10: printf("1.0b\n"); break; case HCI_SPEC_V11: printf("1.1\n"); break; case HCI_SPEC_V12: printf("1.2\n"); break; case HCI_SPEC_V20: printf("2.0 + EDR\n"); break; case HCI_SPEC_V21: printf("2.1 + EDR\n"); break; case HCI_SPEC_V30: printf("3.0 + HS\n"); break; default: printf("unknown\n"); break; } load_value(HCI_CMD_READ_UNIT_CLASS, buf, HCI_CLASS_SIZE); print_class("\tclass:", buf); load_value(HCI_CMD_READ_LOCAL_NAME, buf, HCI_UNIT_NAME_SIZE); printf("\tname: \"%s\"\n", buf); load_value(HCI_CMD_READ_VOICE_SETTING, buf, sizeof(uint16_t)); voice = (buf[1] << 8) | buf[0]; print_voice(level); load_value(HCI_CMD_READ_PIN_TYPE, &val, sizeof(val)); printf("\tpin: %s\n", val ? "fixed" : "variable"); val = 0; if (version >= HCI_SPEC_V12) load_value(HCI_CMD_READ_INQUIRY_MODE, &val, sizeof(val)); print_val("inquiry mode", imodes, val); width = printf("\toptions:"); load_value(HCI_CMD_READ_SCAN_ENABLE, &val, sizeof(val)); if (val & HCI_INQUIRY_SCAN_ENABLE) tag("iscan"); else if (level > 0) tag("-iscan"); if (val & HCI_PAGE_SCAN_ENABLE) tag("pscan"); else if (level > 0) tag("-pscan"); load_value(HCI_CMD_READ_AUTH_ENABLE, &val, sizeof(val)); if (val) tag("auth"); else if (level > 0) tag("-auth"); load_value(HCI_CMD_READ_ENCRYPTION_MODE, &val, sizeof(val)); if (val) tag("encrypt"); else if (level > 0) tag("-encrypt"); val = btr.btr_link_policy; if (val & HCI_LINK_POLICY_ENABLE_ROLE_SWITCH) tag("switch"); else if (level > 0) tag("-switch"); if (val & HCI_LINK_POLICY_ENABLE_HOLD_MODE) tag("hold"); else if (level > 0) tag("-hold"); if (val & HCI_LINK_POLICY_ENABLE_SNIFF_MODE) tag("sniff"); else if (level > 0) tag("-sniff"); if (val & HCI_LINK_POLICY_ENABLE_PARK_MODE) tag("park"); else if (level > 0) tag("-park"); tag(NULL); if (level-- < 1) return; ptype = btr.btr_packet_type; width = printf("\tptype: [0x%04x]", ptype); if (ptype & HCI_PKT_DM1) tag("DM1"); if (ptype & HCI_PKT_DH1) tag("DH1"); if (ptype & HCI_PKT_DM3) tag("DM3"); if (ptype & HCI_PKT_DH3) tag("DH3"); if (ptype & HCI_PKT_DM5) tag("DM5"); if (ptype & HCI_PKT_DH5) tag("DH5"); if ((ptype & HCI_PKT_2MBPS_DH1) == 0) tag("2-DH1"); if ((ptype & HCI_PKT_3MBPS_DH1) == 0) tag("3-DH1"); if ((ptype & HCI_PKT_2MBPS_DH3) == 0) tag("2-DH3"); if ((ptype & HCI_PKT_3MBPS_DH3) == 0) tag("3-DH3"); if ((ptype & HCI_PKT_2MBPS_DH5) == 0) tag("2-DH5"); if ((ptype & HCI_PKT_3MBPS_DH5) == 0) tag("3-DH5"); tag(NULL); load_value(HCI_CMD_READ_PAGE_TIMEOUT, buf, sizeof(uint16_t)); printf("\tpage timeout: %d ms\n", le16dec(buf) * 5 / 8); if (level-- < 1) return; load_value(HCI_CMD_READ_LOCAL_FEATURES, buf, HCI_FEATURES_SIZE); if ((buf[7] & HCI_LMP_EXTENDED_FEATURES) == 0) { print_features("\tfeatures:", 0, buf); } else { hci_read_local_extended_features_rp rp; rp.page = 0; do { hci_req(HCI_CMD_READ_LOCAL_EXTENDED_FEATURES, 0, &rp.page, sizeof(rp.page), &rp, sizeof(rp)); print_features("\tfeatures (page %d):", rp.page, rp.features); } while (rp.page++ < rp.max_page); } } void print_stats(void) { if (sflag == 0) return; if (sflag == 1) { if (ioctl(hci, SIOCGBTSTATS, &btr) < 0) err(EXIT_FAILURE, "SIOCGBTSTATS"); } else { if (ioctl(hci, SIOCZBTSTATS, &btr) < 0) err(EXIT_FAILURE, "SIOCZBTSTATS"); } printf( "\tTotal bytes sent %d, recieved %d\n" "\tCommands sent %d, Events received %d\n" "\tACL data packets sent %d, received %d\n" "\tSCO data packets sent %d, received %d\n" "\tInput errors %d, Output errors %d\n", btr.btr_stats.byte_tx, btr.btr_stats.byte_rx, btr.btr_stats.cmd_tx, btr.btr_stats.evt_rx, btr.btr_stats.acl_tx, btr.btr_stats.acl_rx, btr.btr_stats.sco_tx, btr.btr_stats.sco_rx, btr.btr_stats.err_rx, btr.btr_stats.err_tx); } void print_features(const char *fmt, uint8_t page, uint8_t *f) { width = printf(fmt, page); switch(page) { case 0: print_features0(f); break; case 1: print_features1(f); break; default: break; } tag(NULL); } void print_features0(uint8_t *f) { /* ------------------- byte 0 --------------------*/ if (*f & HCI_LMP_3SLOT) tag("<3 slot>"); if (*f & HCI_LMP_5SLOT) tag("<5 slot>"); if (*f & HCI_LMP_ENCRYPTION) tag(""); if (*f & HCI_LMP_SLOT_OFFSET) tag(""); if (*f & HCI_LMP_TIMIACCURACY) tag(""); if (*f & HCI_LMP_ROLE_SWITCH) tag(""); if (*f & HCI_LMP_HOLD_MODE) tag(""); if (*f & HCI_LMP_SNIFF_MODE) tag(""); f++; /* ------------------- byte 1 --------------------*/ if (*f & HCI_LMP_PARK_MODE) tag(""); if (*f & HCI_LMP_RSSI) tag(""); if (*f & HCI_LMP_CHANNEL_QUALITY) tag(""); if (*f & HCI_LMP_SCO_LINK) tag(""); if (*f & HCI_LMP_HV2_PKT) tag(""); if (*f & HCI_LMP_HV3_PKT) tag(""); if (*f & HCI_LMP_ULAW_LOG) tag(""); if (*f & HCI_LMP_ALAW_LOG) tag(""); f++; /* ------------------- byte 1 --------------------*/ if (*f & HCI_LMP_CVSD) tag(""); if (*f & HCI_LMP_PAGISCHEME) tag(""); if (*f & HCI_LMP_POWER_CONTROL) tag(""); if (*f & HCI_LMP_TRANSPARENT_SCO) tag(""); if (*f & HCI_LMP_FLOW_CONTROL_LAG0) tag(""); if (*f & HCI_LMP_FLOW_CONTROL_LAG1) tag(""); if (*f & HCI_LMP_FLOW_CONTROL_LAG2) tag(""); if (*f & HCI_LMP_BC_ENCRYPTION) tag(""); f++; /* ------------------- byte 3 --------------------*/ if (*f & HCI_LMP_EDR_ACL_2MBPS) tag(""); if (*f & HCI_LMP_EDR_ACL_3MBPS) tag(""); if (*f & HCI_LMP_ENHANCED_ISCAN) tag(""); if (*f & HCI_LMP_INTERLACED_ISCAN) tag(""); if (*f & HCI_LMP_INTERLACED_PSCAN) tag(""); if (*f & HCI_LMP_RSSI_INQUIRY) tag(""); if (*f & HCI_LMP_EV3_PKT) tag(""); f++; /* ------------------- byte 4 --------------------*/ if (*f & HCI_LMP_EV4_PKT) tag(""); if (*f & HCI_LMP_EV5_PKT) tag(""); if (*f & HCI_LMP_AFH_CAPABLE_SLAVE) tag(""); if (*f & HCI_LMP_AFH_CLASS_SLAVE) tag(""); if (*f & HCI_LMP_3SLOT_EDR_ACL) tag("<3 slot EDR ACL>"); f++; /* ------------------- byte 5 --------------------*/ if (*f & HCI_LMP_5SLOT_EDR_ACL) tag("<5 slot EDR ACL>"); if (*f & HCI_LMP_SNIFF_SUBRATING) tag(""); if (*f & HCI_LMP_PAUSE_ENCRYPTION) tag(""); if (*f & HCI_LMP_AFH_CAPABLE_MASTER)tag(""); if (*f & HCI_LMP_AFH_CLASS_MASTER) tag(""); if (*f & HCI_LMP_EDR_eSCO_2MBPS) tag(""); if (*f & HCI_LMP_EDR_eSCO_3MBPS) tag(""); if (*f & HCI_LMP_3SLOT_EDR_eSCO) tag("<3 slot EDR eSCO>"); f++; /* ------------------- byte 6 --------------------*/ if (*f & HCI_LMP_EXTENDED_INQUIRY) tag(""); if (*f & HCI_LMP_SIMPLE_PAIRING) tag(""); if (*f & HCI_LMP_ENCAPSULATED_PDU) tag(""); if (*f & HCI_LMP_ERRDATA_REPORTING) tag(""); if (*f & HCI_LMP_NOFLUSH_PB_FLAG) tag(""); f++; /* ------------------- byte 7 --------------------*/ if (*f & HCI_LMP_LINK_SUPERVISION_TO)tag(""); if (*f & HCI_LMP_INQ_RSP_TX_POWER) tag(""); if (*f & HCI_LMP_ENHANCED_POWER_CONTROL)tag(""); if (*f & HCI_LMP_EXTENDED_FEATURES) tag(""); } void print_features1(uint8_t *f) { /* ------------------- byte 0 --------------------*/ if (*f & HCI_LMP_SSP) tag(""); } void print_class(const char *str, uint8_t *uclass) { class = (uclass[2] << 16) | (uclass[1] << 8) | uclass[0]; width = printf("%s [0x%06x]", str, class); switch(__SHIFTOUT(class, __BITS(0, 1))) { case 0: print_class0(); break; default: break; } tag(NULL); } void print_class0(void) { switch (__SHIFTOUT(class, __BITS(8, 12))) { case 1: /* Computer */ switch (__SHIFTOUT(class, __BITS(2, 7))) { case 1: tag("Desktop workstation"); break; case 2: tag("Server-class computer"); break; case 3: tag("Laptop"); break; case 4: tag("Handheld PC/PDA"); break; case 5: tag("Palm Sized PC/PDA"); break; case 6: tag("Wearable computer"); break; default: tag("Computer"); break; } break; case 2: /* Phone */ switch (__SHIFTOUT(class, __BITS(2, 7))) { case 1: tag("Cellular Phone"); break; case 2: tag("Cordless Phone"); break; case 3: tag("Smart Phone"); break; case 4: tag("Wired Modem/Phone Gateway"); break; case 5: tag("Common ISDN"); break; default:tag("Phone"); break; } break; case 3: /* LAN */ tag("LAN"); switch (__SHIFTOUT(class, __BITS(5, 7))) { case 0: tag("[Fully available]"); break; case 1: tag("[1-17% utilised]"); break; case 2: tag("[17-33% utilised]"); break; case 3: tag("[33-50% utilised]"); break; case 4: tag("[50-67% utilised]"); break; case 5: tag("[67-83% utilised]"); break; case 6: tag("[83-99% utilised]"); break; case 7: tag("[No service available]"); break; } break; case 4: /* Audio/Visual */ switch (__SHIFTOUT(class, __BITS(2, 7))) { case 1: tag("Wearable Headset"); break; case 2: tag("Hands-free Audio"); break; case 4: tag("Microphone"); break; case 5: tag("Loudspeaker"); break; case 6: tag("Headphones"); break; case 7: tag("Portable Audio"); break; case 8: tag("Car Audio"); break; case 9: tag("Set-top Box"); break; case 10: tag("HiFi Audio"); break; case 11: tag("VCR"); break; case 12: tag("Video Camera"); break; case 13: tag("Camcorder"); break; case 14: tag("Video Monitor"); break; case 15: tag("Video Display and Loudspeaker"); break; case 16: tag("Video Conferencing"); break; case 18: tag("A/V [Gaming/Toy]"); break; default: tag("Audio/Visual"); break; } break; case 5: /* Peripheral */ switch (__SHIFTOUT(class, __BITS(2, 5))) { case 1: tag("Joystick"); break; case 2: tag("Gamepad"); break; case 3: tag("Remote Control"); break; case 4: tag("Sensing Device"); break; case 5: tag("Digitiser Tablet"); break; case 6: tag("Card Reader"); break; default: tag("Peripheral"); break; } if (class & __BIT(6)) tag("Keyboard"); if (class & __BIT(7)) tag("Mouse"); break; case 6: /* Imaging */ if (class & __BIT(4)) tag("Display"); if (class & __BIT(5)) tag("Camera"); if (class & __BIT(6)) tag("Scanner"); if (class & __BIT(7)) tag("Printer"); if ((class & __BITS(4, 7)) == 0) tag("Imaging"); break; case 7: /* Wearable */ switch (__SHIFTOUT(class, __BITS(2, 7))) { case 1: tag("Wrist Watch"); break; case 2: tag("Pager"); break; case 3: tag("Jacket"); break; case 4: tag("Helmet"); break; case 5: tag("Glasses"); break; default: tag("Wearable"); break; } break; case 8: /* Toy */ switch (__SHIFTOUT(class, __BITS(2, 7))) { case 1: tag("Robot"); break; case 2: tag("Vehicle"); break; case 3: tag("Doll / Action Figure"); break; case 4: tag("Controller"); break; case 5: tag("Game"); break; default: tag("Toy"); break; } break; case 9: /* Health */ switch (__SHIFTOUT(class, __BITS(2, 7))) { case 1: tag("Blood Pressure Monitor"); break; case 2: tag("Thermometer"); break; case 3: tag("Weighing Scale"); break; case 4: tag("Glucose Meter"); break; case 5: tag("Pulse Oximeter"); break; case 6: tag("Heart/Pulse Rate Monitor"); break; case 7: tag("Health Data Display"); break; default: tag("Health"); break; } break; default: break; } if (class & __BIT(13)) tag(""); if (class & __BIT(16)) tag(""); if (class & __BIT(17)) tag(""); if (class & __BIT(18)) tag(""); if (class & __BIT(19)) tag(""); if (class & __BIT(20)) tag(""); if (class & __BIT(21)) tag("