From: Gustavo Padovan <gustavo.padovan@xxxxxxxxxxxxxxx> --- .gitignore | 2 +- Makefile.tools | 8 +- mgmt/main.c | 1933 ----------------------------------------------------- tools/mgmt/main.c | 1933 +++++++++++++++++++++++++++++++++++++++++++++++++++++ 4 files changed, 1938 insertions(+), 1938 deletions(-) delete mode 100644 mgmt/main.c create mode 100644 tools/mgmt/main.c diff --git a/.gitignore b/.gitignore index 7bbefc9..fa1a215 100644 --- a/.gitignore +++ b/.gitignore @@ -84,7 +84,7 @@ compat/dund compat/hidd compat/pand unit/test-eir -mgmt/btmgmt +tools/mgmt/btmgmt monitor/btmon tools/emulator/btvirt diff --git a/Makefile.tools b/Makefile.tools index f7c6f9f..36b7178 100644 --- a/Makefile.tools +++ b/Makefile.tools @@ -50,10 +50,10 @@ tools_ppporc_LDADD = lib/libbluetooth-private.la tools_hcieventmask_LDADD = lib/libbluetooth-private.la -noinst_PROGRAMS += mgmt/btmgmt monitor/btmon tools/emulator/btvirt +noinst_PROGRAMS += tools/mgmt/btmgmt monitor/btmon tools/emulator/btvirt -mgmt_btmgmt_SOURCES = mgmt/main.c src/glib-helper.c -mgmt_btmgmt_LDADD = lib/libbluetooth-private.la @GLIB_LIBS@ +tools_mgmt_btmgmt_SOURCES = tools/mgmt/main.c src/glib-helper.c +tools_mgmt_btmgmt_LDADD = lib/libbluetooth-private.la @GLIB_LIBS@ monitor_btmon_SOURCES = monitor/main.c monitor/bt.h \ monitor/mainloop.h monitor/mainloop.c \ @@ -63,7 +63,7 @@ monitor_btmon_SOURCES = monitor/main.c monitor/bt.h \ monitor/packet.h monitor/packet.c monitor_btmon_LDADD = lib/libbluetooth-private.la -emulator_btvirt_SOURCES = tools/emulator/main.c monitor/bt.h \ +tools_emulator_btvirt_SOURCES = tools/emulator/main.c monitor/bt.h \ monitor/mainloop.h monitor/mainloop.c \ tools/emulator/server.h tools/emulator/server.c \ tools/emulator/vhci.h tools/emulator/vhci.c \ diff --git a/mgmt/main.c b/mgmt/main.c deleted file mode 100644 index b2d6c3c..0000000 --- a/mgmt/main.c +++ /dev/null @@ -1,1933 +0,0 @@ -/* - * BlueZ - Bluetooth protocol stack for Linux - * - * Copyright (C) 2011 Intel Corporation. All rights reserved. - * - * This program is free software; you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation; either version 2 of the License, or - * (at your option) any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program; if not, write to the Free Software - * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA - * - */ - -#ifdef HAVE_CONFIG_H -#include <config.h> -#endif - -#include <stdlib.h> -#include <stdio.h> -#include <unistd.h> -#include <errno.h> -#include <string.h> -#include <sys/types.h> -#include <sys/socket.h> -#include <poll.h> -#include <getopt.h> -#include <stdbool.h> - -#include <bluetooth/bluetooth.h> -#include <bluetooth/hci.h> -#include <bluetooth/hci_lib.h> -#include <bluetooth/sdp.h> -#include <bluetooth/sdp_lib.h> -#include <bluetooth/mgmt.h> - -#include <glib.h> -#include "glib-helper.h" - -static bool monitor = false; -static bool discovery = false; -static bool resolve_names = true; - -typedef void (*cmd_cb)(int mgmt_sk, uint16_t op, uint16_t id, uint8_t status, - void *rsp, uint16_t len, void *user_data); - -static struct pending_cmd { - uint16_t op; - uint16_t id; - cmd_cb cb; - void *user_data; - struct pending_cmd *next; -} *pending = NULL; - -static int mgmt_send_cmd(int mgmt_sk, uint16_t op, uint16_t id, void *data, - size_t len, cmd_cb func, void *user_data) -{ - char buf[1024]; - struct pending_cmd *cmd; - struct mgmt_hdr *hdr = (void *) buf; - - if (len + MGMT_HDR_SIZE > sizeof(buf)) - return -EINVAL; - - cmd = calloc(1, sizeof(struct pending_cmd)); - if (cmd == NULL) - return -errno; - - cmd->op = op; - cmd->id = id; - cmd->cb = func; - cmd->user_data = user_data; - - memset(buf, 0, sizeof(buf)); - hdr->opcode = htobs(op); - hdr->index = htobs(id); - hdr->len = htobs(len); - memcpy(buf + MGMT_HDR_SIZE, data, len); - - if (write(mgmt_sk, buf, MGMT_HDR_SIZE + len) < 0) { - fprintf(stderr, "Unable to write to socket: %s\n", - strerror(errno)); - free(cmd); - return -1; - } - - cmd->next = pending; - pending = cmd; - - return 0; -} - -static int mgmt_open(void) -{ - struct sockaddr_hci addr; - int sk; - - sk = socket(AF_BLUETOOTH, SOCK_RAW, BTPROTO_HCI); - if (sk < 0) { - fprintf(stderr, "socket: %s\n", strerror(errno)); - return sk; - } - - memset(&addr, 0, sizeof(addr)); - addr.hci_family = AF_BLUETOOTH; - addr.hci_dev = HCI_DEV_NONE; - addr.hci_channel = HCI_CHANNEL_CONTROL; - - if (bind(sk, (struct sockaddr *) &addr, sizeof(addr)) < 0) { - fprintf(stderr, "bind: %s\n", strerror(errno)); - close(sk); - return -1; - } - - return sk; -} - -static void mgmt_check_pending(int mgmt_sk, uint16_t op, uint16_t index, - uint16_t status, void *data, uint16_t len) -{ - struct pending_cmd *c, *prev; - - for (c = pending, prev = NULL; c != NULL; prev = c, c = c->next) { - if (c->op != op) - continue; - if (c->id != index) - continue; - - if (c == pending) - pending = c->next; - else - prev->next = c->next; - - c->cb(mgmt_sk, op, index, status, data, len, c->user_data); - - free(c); - break; - } -} - -static int mgmt_cmd_complete(int mgmt_sk, uint16_t index, - struct mgmt_ev_cmd_complete *ev, uint16_t len) -{ - uint16_t op; - - if (len < sizeof(*ev)) { - fprintf(stderr, "Too short (%u bytes) cmd complete event\n", - len); - return -EINVAL; - } - - op = bt_get_le16(&ev->opcode); - - len -= sizeof(*ev); - - if (monitor) - printf("%s complete, opcode 0x%04x len %u\n", mgmt_opstr(op), - op, len); - - mgmt_check_pending(mgmt_sk, op, index, ev->status, ev->data, len); - - return 0; -} - -static int mgmt_cmd_status(int mgmt_sk, uint16_t index, - struct mgmt_ev_cmd_status *ev, uint16_t len) -{ - uint16_t opcode; - - if (len < sizeof(*ev)) { - fprintf(stderr, "Too short (%u bytes) cmd status event\n", - len); - return -EINVAL; - } - - opcode = bt_get_le16(&ev->opcode); - - if (monitor) - printf("cmd status, opcode 0x%04x status 0x%02x (%s)\n", - opcode, ev->status, mgmt_errstr(ev->status)); - - if (ev->status != 0) - mgmt_check_pending(mgmt_sk, opcode, index, ev->status, - NULL, 0); - - return 0; -} - -static int mgmt_controller_error(uint16_t index, - struct mgmt_ev_controller_error *ev, - uint16_t len) -{ - if (len < sizeof(*ev)) { - fprintf(stderr, - "Too short (%u bytes) controller error event\n", len); - return -EINVAL; - } - - if (monitor) - printf("hci%u error 0x%02x\n", index, ev->error_code); - - return 0; -} - -static int mgmt_index_added(int mgmt_sk, uint16_t index) -{ - if (monitor) - printf("hci%u added\n", index); - return 0; -} - -static int mgmt_index_removed(int mgmt_sk, uint16_t index) -{ - if (monitor) - printf("hci%u removed\n", index); - return 0; -} - -static const char *settings_str[] = { - "powered", - "connectable", - "fast-connectable", - "discoverable", - "pairable", - "link-security", - "ssp", - "br/edr", - "hs", - "le" , -}; - -static void print_settings(uint32_t settings) -{ - unsigned i; - - for (i = 0; i < NELEM(settings_str); i++) { - if ((settings & (1 << i)) != 0) - printf("%s ", settings_str[i]); - } -} - -static int mgmt_new_settings(int mgmt_sk, uint16_t index, - uint32_t *ev, uint16_t len) -{ - if (len < sizeof(*ev)) { - fprintf(stderr, "Too short new_settings event (%u)\n", len); - return -EINVAL; - } - - if (monitor) { - printf("hci%u new_settings: ", index); - print_settings(bt_get_le32(ev)); - printf("\n"); - } - - return 0; -} - -static int mgmt_discovering(int mgmt_sk, uint16_t index, - struct mgmt_ev_discovering *ev, uint16_t len) -{ - if (len < sizeof(*ev)) { - fprintf(stderr, "Too short (%u bytes) discovering event\n", - len); - return -EINVAL; - } - - if (ev->discovering == 0 && discovery) - exit(EXIT_SUCCESS); - - if (monitor) - printf("hci%u type %u discovering %s\n", index, - ev->type, ev->discovering ? "on" : "off"); - - return 0; -} - -static int mgmt_new_link_key(int mgmt_sk, uint16_t index, - struct mgmt_ev_new_link_key *ev, uint16_t len) -{ - - if (len != sizeof(*ev)) { - fprintf(stderr, "Invalid new_link_key length (%u bytes)\n", - len); - return -EINVAL; - } - - if (monitor) { - char addr[18]; - ba2str(&ev->key.addr.bdaddr, addr); - printf("hci%u new_link_key %s type 0x%02x pin_len %d " - "store_hint %u\n", index, addr, ev->key.type, - ev->key.pin_len, ev->store_hint); - } - - return 0; -} - -static const char *typestr(uint8_t type) -{ - const char *str[] = { "BR/EDR", "LE Public", "LE Random" }; - - if (type <= BDADDR_LE_RANDOM) - return str[type]; - - return "(unknown)"; -} - -static int mgmt_connected(int mgmt_sk, uint16_t index, - struct mgmt_ev_device_connected *ev, - uint16_t len) -{ - uint16_t eir_len; - - if (len < sizeof(*ev)) { - fprintf(stderr, - "Invalid connected event length (%u bytes)\n", len); - return -EINVAL; - } - - eir_len = bt_get_le16(&ev->eir_len); - if (len != sizeof(*ev) + eir_len) { - fprintf(stderr, "Invalid connected event length " - "(%u bytes, eir_len %u bytes)\n", len, eir_len); - return -EINVAL; - } - - if (monitor) { - char addr[18]; - ba2str(&ev->addr.bdaddr, addr); - printf("hci%u %s type %s connected eir_len %u\n", index, addr, - typestr(ev->addr.type), eir_len); - } - - return 0; -} - -static int mgmt_disconnected(int mgmt_sk, uint16_t index, - struct mgmt_addr_info *ev, uint16_t len) -{ - if (len != sizeof(*ev)) { - fprintf(stderr, - "Invalid disconnected event length (%u bytes)\n", len); - return -EINVAL; - } - - if (monitor) { - char addr[18]; - ba2str(&ev->bdaddr, addr); - printf("hci%u %s type %s disconnected\n", index, addr, - typestr(ev->type)); - } - - return 0; -} - -static int mgmt_conn_failed(int mgmt_sk, uint16_t index, - struct mgmt_ev_connect_failed *ev, - uint16_t len) -{ - if (len != sizeof(*ev)) { - fprintf(stderr, - "Invalid connect_failed event length (%u bytes)\n", len); - return -EINVAL; - } - - if (monitor) { - char addr[18]; - ba2str(&ev->addr.bdaddr, addr); - printf("hci%u %s type %s connect failed (status 0x%02x, %s)\n", - index, addr, typestr(ev->addr.type), ev->status, - mgmt_errstr(ev->status)); - } - - return 0; -} - -static int mgmt_auth_failed(int mgmt_sk, uint16_t index, - struct mgmt_ev_auth_failed *ev, - uint16_t len) -{ - if (len != sizeof(*ev)) { - fprintf(stderr, - "Invalid auth_failed event length (%u bytes)\n", len); - return -EINVAL; - } - - if (monitor) { - char addr[18]; - ba2str(&ev->addr.bdaddr, addr); - printf("hci%u %s auth failed with status 0x%02x (%s)\n", - index, addr, ev->status, mgmt_errstr(ev->status)); - } - - return 0; -} - -static int mgmt_name_changed(int mgmt_sk, uint16_t index, - struct mgmt_ev_local_name_changed *ev, - uint16_t len) -{ - if (len != sizeof(*ev)) { - fprintf(stderr, - "Invalid local_name_changed length (%u bytes)\n", len); - return -EINVAL; - } - - if (monitor) - printf("hci%u name changed: %s\n", index, ev->name); - - return 0; -} - -static void confirm_name_rsp(int mgmt_sk, uint16_t op, uint16_t id, - uint8_t status, void *rsp, uint16_t len, - void *user_data) -{ - struct mgmt_rp_confirm_name *rp = rsp; - char addr[18]; - - if (len == 0 && status != 0) { - fprintf(stderr, - "hci%u confirm_name failed with status 0x%02x (%s)\n", - id, status, mgmt_errstr(status)); - return; - } - - if (len != sizeof(*rp)) { - fprintf(stderr, - "hci%u confirm_name rsp length %u instead of %zu\n", - id, len, sizeof(*rp)); - return; - } - - ba2str(&rp->addr.bdaddr, addr); - - if (status != 0) - fprintf(stderr, - "hci%u confirm_name for %s failed: 0x%02x (%s)\n", - id, addr, status, mgmt_errstr(status)); - else - printf("hci%u confirm_name succeeded for %s\n", id, addr); -} - -static int mgmt_device_found(int mgmt_sk, uint16_t index, - struct mgmt_ev_device_found *ev, uint16_t len) -{ - uint32_t flags; - uint16_t eir_len; - - if (len < sizeof(*ev)) { - fprintf(stderr, - "Too short device_found length (%u bytes)\n", len); - return -EINVAL; - } - - flags = btohs(ev->flags); - - eir_len = bt_get_le16(&ev->eir_len); - if (len != sizeof(*ev) + eir_len) { - fprintf(stderr, "dev_found: expected %zu bytes, got %u bytes", - sizeof(*ev) + eir_len, len); - return -EINVAL; - } - - if (monitor || discovery) { - char addr[18]; - ba2str(&ev->addr.bdaddr, addr); - printf("hci%u dev_found: %s type %s rssi %d " - "flags 0x%04x eir_len %u\n", index, addr, - typestr(ev->addr.type), ev->rssi, flags, eir_len); - } - - if (discovery && (flags & MGMT_DEV_FOUND_CONFIRM_NAME)) { - struct mgmt_cp_confirm_name cp; - - memset(&cp, 0, sizeof(cp)); - memcpy(&cp.addr, &ev->addr, sizeof(cp.addr)); - if (resolve_names) - cp.name_known = 0; - else - cp.name_known = 1; - - mgmt_send_cmd(mgmt_sk, MGMT_OP_CONFIRM_NAME, index, - &cp, sizeof(cp), confirm_name_rsp, - NULL); - } - - return 0; -} - -static void pin_rsp(int mgmt_sk, uint16_t op, uint16_t id, uint8_t status, - void *rsp, uint16_t len, void *user_data) -{ - if (status != 0) { - fprintf(stderr, - "hci%u PIN Code reply failed with status 0x%02x (%s)", - id, status, mgmt_errstr(status)); - exit(EXIT_FAILURE); - } - - printf("hci%u PIN Reply successful\n", id); -} - -static int mgmt_pin_reply(int mgmt_sk, uint16_t index, - struct mgmt_addr_info *addr, - const char *pin, size_t len) -{ - struct mgmt_cp_pin_code_reply cp; - - memset(&cp, 0, sizeof(cp)); - memcpy(&cp.addr, addr, sizeof(cp.addr)); - cp.pin_len = len; - memcpy(cp.pin_code, pin, len); - - return mgmt_send_cmd(mgmt_sk, MGMT_OP_PIN_CODE_REPLY, index, - &cp, sizeof(cp), pin_rsp, NULL); -} - -static void pin_neg_rsp(int mgmt_sk, uint16_t op, uint16_t id, uint8_t status, - void *rsp, uint16_t len, void *user_data) -{ - if (status != 0) { - fprintf(stderr, - "hci%u PIN Neg reply failed with status 0x%02x (%s)", - id, status, mgmt_errstr(status)); - exit(EXIT_FAILURE); - } - - printf("hci%u PIN Negative Reply successful\n", id); -} - -static int mgmt_pin_neg_reply(int mgmt_sk, uint16_t index, - struct mgmt_addr_info *addr) -{ - struct mgmt_cp_pin_code_neg_reply cp; - - memset(&cp, 0, sizeof(cp)); - memcpy(&cp.addr, addr, sizeof(cp.addr)); - - return mgmt_send_cmd(mgmt_sk, MGMT_OP_PIN_CODE_NEG_REPLY, index, - &cp, sizeof(cp), pin_neg_rsp, NULL); -} - -static int mgmt_request_pin(int mgmt_sk, uint16_t index, - struct mgmt_ev_pin_code_request *ev, - uint16_t len) -{ - char pin[18]; - size_t pin_len; - - if (len != sizeof(*ev)) { - fprintf(stderr, - "Invalid pin_code request length (%u bytes)\n", len); - return -EINVAL; - } - - if (monitor) { - char addr[18]; - ba2str(&ev->addr.bdaddr, addr); - printf("hci%u %s request PIN\n", index, addr); - } - - printf("PIN Request (press enter to reject) >> "); - fflush(stdout); - - memset(pin, 0, sizeof(pin)); - - if (fgets(pin, sizeof(pin), stdin) == NULL || pin[0] == '\n') - return mgmt_pin_neg_reply(mgmt_sk, index, &ev->addr); - - pin_len = strlen(pin); - if (pin[pin_len - 1] == '\n') { - pin[pin_len - 1] = '\0'; - pin_len--; - } - - return mgmt_pin_reply(mgmt_sk, index, &ev->addr, pin, pin_len); -} - -static void confirm_rsp(int mgmt_sk, uint16_t op, uint16_t id, uint8_t status, - void *rsp, uint16_t len, void *user_data) -{ - if (status != 0) { - fprintf(stderr, - "hci%u User Confirm reply failed. status 0x%02x (%s)", - id, status, mgmt_errstr(status)); - exit(EXIT_FAILURE); - } - - printf("hci%u User Confirm Reply successful\n", id); -} - -static int mgmt_confirm_reply(int mgmt_sk, uint16_t index, bdaddr_t *bdaddr) -{ - struct mgmt_cp_user_confirm_reply cp; - - memset(&cp, 0, sizeof(cp)); - bacpy(&cp.addr.bdaddr, bdaddr); - - return mgmt_send_cmd(mgmt_sk, MGMT_OP_USER_CONFIRM_REPLY, index, - &cp, sizeof(cp), confirm_rsp, NULL); -} - -static void confirm_neg_rsp(int mgmt_sk, uint16_t op, uint16_t id, - uint8_t status, void *rsp, uint16_t len, - void *user_data) -{ - if (status != 0) { - fprintf(stderr, - "hci%u Confirm Neg reply failed. status 0x%02x (%s)", - id, status, mgmt_errstr(status)); - exit(EXIT_FAILURE); - } - - printf("hci%u User Confirm Negative Reply successful\n", id); -} - -static int mgmt_confirm_neg_reply(int mgmt_sk, uint16_t index, - bdaddr_t *bdaddr) -{ - struct mgmt_cp_user_confirm_reply cp; - - memset(&cp, 0, sizeof(cp)); - bacpy(&cp.addr.bdaddr, bdaddr); - - return mgmt_send_cmd(mgmt_sk, MGMT_OP_USER_CONFIRM_NEG_REPLY, index, - &cp, sizeof(cp), confirm_neg_rsp, NULL); -} - - -static int mgmt_user_confirm(int mgmt_sk, uint16_t index, - struct mgmt_ev_user_confirm_request *ev, - uint16_t len) -{ - char rsp[5]; - size_t rsp_len; - uint32_t val; - char addr[18]; - - if (len != sizeof(*ev)) { - fprintf(stderr, - "Invalid user_confirm request length (%u)\n", len); - return -EINVAL; - } - - ba2str(&ev->addr.bdaddr, addr); - val = bt_get_le32(&ev->value); - - if (monitor) - printf("hci%u %s User Confirm %06u hint %u\n", index, addr, - val, ev->confirm_hint); - - if (ev->confirm_hint) - printf("Accept pairing with %s (yes/no) >> ", addr); - else - printf("Confirm value %06u for %s (yes/no) >> ", val, addr); - - fflush(stdout); - - memset(rsp, 0, sizeof(rsp)); - - if (fgets(rsp, sizeof(rsp), stdin) == NULL || rsp[0] == '\n') - return mgmt_confirm_neg_reply(mgmt_sk, index, &ev->addr.bdaddr); - - rsp_len = strlen(rsp); - if (rsp[rsp_len - 1] == '\n') { - rsp[rsp_len - 1] = '\0'; - rsp_len--; - } - - if (rsp[0] == 'y' || rsp[0] == 'Y') - return mgmt_confirm_reply(mgmt_sk, index, &ev->addr.bdaddr); - else - return mgmt_confirm_neg_reply(mgmt_sk, index, &ev->addr.bdaddr); -} - -static int mgmt_handle_event(int mgmt_sk, uint16_t ev, uint16_t index, - void *data, uint16_t len) -{ - if (monitor) - printf("event: %s\n", mgmt_evstr(ev)); - - switch (ev) { - case MGMT_EV_CMD_COMPLETE: - return mgmt_cmd_complete(mgmt_sk, index, data, len); - case MGMT_EV_CMD_STATUS: - return mgmt_cmd_status(mgmt_sk, index, data, len); - case MGMT_EV_CONTROLLER_ERROR: - return mgmt_controller_error(index, data, len); - case MGMT_EV_INDEX_ADDED: - return mgmt_index_added(mgmt_sk, index); - case MGMT_EV_INDEX_REMOVED: - return mgmt_index_removed(mgmt_sk, index); - case MGMT_EV_NEW_SETTINGS: - return mgmt_new_settings(mgmt_sk, index, data, len); - case MGMT_EV_DISCOVERING: - return mgmt_discovering(mgmt_sk, index, data, len); - case MGMT_EV_NEW_LINK_KEY: - return mgmt_new_link_key(mgmt_sk, index, data, len); - case MGMT_EV_DEVICE_CONNECTED: - return mgmt_connected(mgmt_sk, index, data, len); - case MGMT_EV_DEVICE_DISCONNECTED: - return mgmt_disconnected(mgmt_sk, index, data, len); - case MGMT_EV_CONNECT_FAILED: - return mgmt_conn_failed(mgmt_sk, index, data, len); - case MGMT_EV_AUTH_FAILED: - return mgmt_auth_failed(mgmt_sk, index, data, len); - case MGMT_EV_LOCAL_NAME_CHANGED: - return mgmt_name_changed(mgmt_sk, index, data, len); - case MGMT_EV_DEVICE_FOUND: - return mgmt_device_found(mgmt_sk, index, data, len); - case MGMT_EV_PIN_CODE_REQUEST: - return mgmt_request_pin(mgmt_sk, index, data, len); - case MGMT_EV_USER_CONFIRM_REQUEST: - return mgmt_user_confirm(mgmt_sk, index, data, len); - default: - if (monitor) - printf("Unhandled event 0x%04x (%s)\n", ev, mgmt_evstr(ev)); - return 0; - } -} - -static int mgmt_process_data(int mgmt_sk) -{ - char buf[1024]; - struct mgmt_hdr *hdr = (void *) buf; - uint16_t len, ev, index; - ssize_t ret; - - ret = read(mgmt_sk, buf, sizeof(buf)); - if (ret < 0) { - fprintf(stderr, "read: %s\n", strerror(errno)); - return ret; - } - - if (ret < MGMT_HDR_SIZE) { - fprintf(stderr, "Too small mgmt packet (%zd bytes)\n", ret); - return 0; - } - - ev = bt_get_le16(&hdr->opcode); - index = bt_get_le16(&hdr->index); - len = bt_get_le16(&hdr->len); - - if (monitor) - printf("event 0x%04x len 0x%04x index 0x%04x\n", ev, len, index); - - if (ret != MGMT_HDR_SIZE + len) { - fprintf(stderr, "Packet length mismatch. ret %zd len %u", - ret, len); - return 0; - } - - mgmt_handle_event(mgmt_sk, ev, index, buf + MGMT_HDR_SIZE, len); - - return 0; -} - -static void cmd_monitor(int mgmt_sk, uint16_t index, int argc, char **argv) -{ - printf("Monitoring mgmt events...\n"); - monitor = true; -} - -static void version_rsp(int mgmt_sk, uint16_t op, uint16_t id, uint8_t status, - void *rsp, uint16_t len, void *user_data) -{ - struct mgmt_rp_read_version *rp = rsp; - - if (status != 0) { - fprintf(stderr, "Reading mgmt version failed with status" - " 0x%02x (%s)\n", status, mgmt_errstr(status)); - exit(EXIT_FAILURE); - } - - if (len < sizeof(*rp)) { - fprintf(stderr, "Too small version reply (%u bytes)\n", len); - exit(EXIT_FAILURE); - } - - printf("MGMT Version %u, revision %u\n", rp->version, - bt_get_le16(&rp->revision)); - - exit(EXIT_SUCCESS); -} - -static void cmd_version(int mgmt_sk, uint16_t index, int argc, char **argv) -{ - if (mgmt_send_cmd(mgmt_sk, MGMT_OP_READ_VERSION, MGMT_INDEX_NONE, - NULL, 0, version_rsp, NULL) < 0) { - fprintf(stderr, "Unable to send read_version cmd\n"); - exit(EXIT_FAILURE); - } -} - -static void commands_rsp(int mgmt_sk, uint16_t op, uint16_t id, uint8_t status, - void *rsp, uint16_t len, void *user_data) -{ - struct mgmt_rp_read_commands *rp = rsp; - uint16_t num_commands, num_events, *opcode; - size_t expected_len; - int i; - - if (status != 0) { - fprintf(stderr, "Reading supported commands failed with status" - " 0x%02x (%s)\n", status, mgmt_errstr(status)); - exit(EXIT_FAILURE); - } - - if (len < sizeof(*rp)) { - fprintf(stderr, "Too small commands reply (%u bytes)\n", len); - exit(EXIT_FAILURE); - } - - num_commands = bt_get_le16(&rp->num_commands); - num_events = bt_get_le16(&rp->num_events); - - expected_len = sizeof(*rp) + num_commands * sizeof(uint16_t) + - num_events * sizeof(uint16_t); - - if (len < expected_len) { - fprintf(stderr, "Too small commands reply (%u != %zu)\n", - len, expected_len); - exit(EXIT_FAILURE); - } - - opcode = rp->opcodes; - - printf("%u commands:\n", num_commands); - for (i = 0; i < num_commands; i++) { - uint16_t op = bt_get_le16(opcode++); - printf("\t%s (0x%04x)\n", mgmt_opstr(op), op); - } - - printf("%u events:\n", num_events); - for (i = 0; i < num_events; i++) { - uint16_t ev = bt_get_le16(opcode++); - printf("\t%s (0x%04x)\n", mgmt_evstr(ev), ev); - } - - exit(EXIT_SUCCESS); -} - -static void cmd_commands(int mgmt_sk, uint16_t index, int argc, char **argv) -{ - if (mgmt_send_cmd(mgmt_sk, MGMT_OP_READ_COMMANDS, MGMT_INDEX_NONE, - NULL, 0, commands_rsp, NULL) < 0) { - fprintf(stderr, "Unable to send read_commands cmd\n"); - exit(EXIT_FAILURE); - } -} - -static void info_rsp(int mgmt_sk, uint16_t op, uint16_t id, uint8_t status, - void *rsp, uint16_t len, void *user_data) -{ - struct mgmt_rp_read_info *rp = rsp; - char addr[18]; - - if (status != 0) { - fprintf(stderr, - "Reading hci%u info failed with status 0x%02x (%s)\n", - id, status, mgmt_errstr(status)); - exit(EXIT_FAILURE); - } - - if (len < sizeof(*rp)) { - fprintf(stderr, "Too small info reply (%u bytes)\n", len); - exit(EXIT_FAILURE); - } - - ba2str(&rp->bdaddr, addr); - printf("hci%u:\taddr %s version %u manufacturer %u" - " class 0x%02x%02x%02x\n", - id, addr, rp->version, bt_get_le16(&rp->manufacturer), - rp->dev_class[2], rp->dev_class[1], rp->dev_class[0]); - - printf("\tsupported settings: "); - print_settings(bt_get_le32(&rp->supported_settings)); - - printf("\n\tcurrent settings: "); - print_settings(bt_get_le32(&rp->current_settings)); - - printf("\n\tname %s\n", rp->name); - printf("\tshort name %s\n", rp->short_name); - - if (pending == NULL) - exit(EXIT_SUCCESS); -} - -static void index_rsp(int mgmt_sk, uint16_t op, uint16_t id, uint8_t status, - void *rsp, uint16_t len, void *user_data) -{ - struct mgmt_rp_read_index_list *rp = rsp; - uint16_t count; - unsigned int i; - - if (status != 0) { - fprintf(stderr, - "Reading index list failed with status 0x%02x (%s)\n", - status, mgmt_errstr(status)); - exit(EXIT_FAILURE); - } - - if (len < sizeof(*rp)) { - fprintf(stderr, "Too small index list reply (%u bytes)\n", - len); - exit(EXIT_FAILURE); - } - - count = bt_get_le16(&rp->num_controllers); - - if (len < sizeof(*rp) + count * sizeof(uint16_t)) { - fprintf(stderr, - "Index count (%u) doesn't match reply length (%u)\n", - count, len); - exit(EXIT_FAILURE); - } - - if (monitor) - printf("Index list with %u item%s\n", - count, count > 1 ? "s" : ""); - - if (count == 0) - exit(EXIT_SUCCESS); - - if (monitor && count > 0) - printf("\t"); - - for (i = 0; i < count; i++) { - uint16_t index; - - index = bt_get_le16(&rp->index[i]); - - if (monitor) - printf("hci%u ", index); - - if (mgmt_send_cmd(mgmt_sk, MGMT_OP_READ_INFO, index, NULL, - 0, info_rsp, NULL) < 0) { - fprintf(stderr, "Unable to send read_info cmd\n"); - exit(EXIT_FAILURE); - } - } - - if (monitor && count > 0) - printf("\n"); -} - -static void cmd_info(int mgmt_sk, uint16_t index, int argc, char **argv) -{ - if (index == MGMT_INDEX_NONE) { - if (mgmt_send_cmd(mgmt_sk, MGMT_OP_READ_INDEX_LIST, - MGMT_INDEX_NONE, NULL, 0, - index_rsp, NULL) < 0) { - fprintf(stderr, "Unable to send index_list cmd\n"); - exit(EXIT_FAILURE); - } - - return; - } - - if (mgmt_send_cmd(mgmt_sk, MGMT_OP_READ_INFO, index, NULL, - 0, info_rsp, NULL) < 0) { - fprintf(stderr, "Unable to send read_info cmd\n"); - exit(EXIT_FAILURE); - } -} - -static void setting_rsp(int mgmt_sk, uint16_t op, uint16_t id, uint8_t status, - void *rsp, uint16_t len, void *user_data) -{ - uint32_t *rp = rsp; - - if (status != 0) { - fprintf(stderr, - "%s for hci%u failed with status 0x%02x (%s)\n", - mgmt_opstr(op), id, status, mgmt_errstr(status)); - exit(EXIT_FAILURE); - } - - if (len < sizeof(*rp)) { - fprintf(stderr, "Too small %s response (%u bytes)\n", - mgmt_opstr(op), len); - exit(EXIT_FAILURE); - } - - printf("hci%u %s complete, settings: ", id, mgmt_opstr(op)); - print_settings(bt_get_le32(rp)); - printf("\n"); - - exit(EXIT_SUCCESS); -} - -static void cmd_setting(int mgmt_sk, uint16_t index, uint16_t op, - int argc, char **argv) -{ - uint8_t val; - - if (argc < 2) { - printf("Specify \"on\" or \"off\"\n"); - exit(EXIT_FAILURE); - } - - if (strcasecmp(argv[1], "on") == 0 || strcasecmp(argv[1], "yes") == 0) - val = 1; - else if (strcasecmp(argv[1], "off") == 0) - val = 0; - else - val = atoi(argv[1]); - - if (index == MGMT_INDEX_NONE) - index = 0; - - if (mgmt_send_cmd(mgmt_sk, op, index, &val, sizeof(val), - setting_rsp, NULL) < 0) { - fprintf(stderr, "Unable to send %s cmd\n", mgmt_opstr(op)); - exit(EXIT_FAILURE); - } -} - -static void cmd_power(int mgmt_sk, uint16_t index, int argc, char **argv) -{ - cmd_setting(mgmt_sk, index, MGMT_OP_SET_POWERED, argc, argv); -} - -static void cmd_discov(int mgmt_sk, uint16_t index, int argc, char **argv) -{ - struct mgmt_cp_set_discoverable cp; - - if (argc < 2) { - printf("Usage: btmgmt %s <yes/no> [timeout]\n", argv[0]); - exit(EXIT_FAILURE); - } - - memset(&cp, 0, sizeof(cp)); - - if (strcasecmp(argv[1], "on") == 0 || strcasecmp(argv[1], "yes") == 0) - cp.val = 1; - else if (strcasecmp(argv[1], "off") == 0) - cp.val = 0; - else - cp.val = atoi(argv[1]); - - if (argc > 2) - cp.timeout = htobs(atoi(argv[2])); - - if (index == MGMT_INDEX_NONE) - index = 0; - - if (mgmt_send_cmd(mgmt_sk, MGMT_OP_SET_DISCOVERABLE, index, - &cp, sizeof(cp), setting_rsp, NULL) < 0) { - fprintf(stderr, "Unable to send set_discoverable cmd\n"); - exit(EXIT_FAILURE); - } -} - -static void cmd_connectable(int mgmt_sk, uint16_t index, int argc, char **argv) -{ - cmd_setting(mgmt_sk, index, MGMT_OP_SET_CONNECTABLE, argc, argv); -} - -static void cmd_pairable(int mgmt_sk, uint16_t index, int argc, char **argv) -{ - cmd_setting(mgmt_sk, index, MGMT_OP_SET_PAIRABLE, argc, argv); -} - -static void cmd_linksec(int mgmt_sk, uint16_t index, int argc, char **argv) -{ - cmd_setting(mgmt_sk, index, MGMT_OP_SET_LINK_SECURITY, argc, argv); -} - -static void cmd_ssp(int mgmt_sk, uint16_t index, int argc, char **argv) -{ - cmd_setting(mgmt_sk, index, MGMT_OP_SET_SSP, argc, argv); -} - -static void cmd_hs(int mgmt_sk, uint16_t index, int argc, char **argv) -{ - cmd_setting(mgmt_sk, index, MGMT_OP_SET_HS, argc, argv); -} - -static void cmd_le(int mgmt_sk, uint16_t index, int argc, char **argv) -{ - cmd_setting(mgmt_sk, index, MGMT_OP_SET_LE, argc, argv); -} - -static void class_rsp(int mgmt_sk, uint16_t op, uint16_t id, uint8_t status, - void *rsp, uint16_t len, void *user_data) -{ - struct mgmt_ev_class_of_dev_changed *rp = rsp; - - if (len == 0 && status != 0) { - fprintf(stderr, "%s failed, status 0x%02x (%s)\n", - mgmt_opstr(op), status, mgmt_errstr(status)); - exit(EXIT_FAILURE); - } - - if (len != sizeof(*rp)) { - fprintf(stderr, "Unexpected %s len %u\n", mgmt_opstr(op), len); - exit(EXIT_FAILURE); - } - - printf("%s succeeded. Class 0x%02x%02x%02x\n", mgmt_opstr(op), - rp->class_of_dev[2], rp->class_of_dev[1], rp->class_of_dev[0]); - - exit(EXIT_SUCCESS); -} - -static void cmd_class(int mgmt_sk, uint16_t index, int argc, char **argv) -{ - uint8_t class[2]; - - if (argc < 3) { - printf("Usage: btmgmt %s <major> <minor>\n", argv[0]); - exit(EXIT_FAILURE); - } - - class[0] = atoi(argv[1]); - class[1] = atoi(argv[2]); - - if (index == MGMT_INDEX_NONE) - index = 0; - - if (mgmt_send_cmd(mgmt_sk, MGMT_OP_SET_DEV_CLASS, index, - class, sizeof(class), class_rsp, NULL) < 0) { - fprintf(stderr, "Unable to send set_dev_class cmd\n"); - exit(EXIT_FAILURE); - } -} - -static void disconnect_rsp(int mgmt_sk, uint16_t op, uint16_t id, - uint8_t status, void *rsp, uint16_t len, - void *user_data) -{ - struct mgmt_rp_disconnect *rp = rsp; - char addr[18]; - - if (len == 0 && status != 0) { - fprintf(stderr, "Disconnect failed with status 0x%02x (%s)\n", - status, mgmt_errstr(status)); - exit(EXIT_FAILURE); - } - - if (len != sizeof(*rp)) { - fprintf(stderr, "Invalid disconnect response length (%u)\n", - len); - exit(EXIT_FAILURE); - } - - ba2str(&rp->addr.bdaddr, addr); - - if (status == 0) { - printf("%s disconnected\n", addr); - exit(EXIT_SUCCESS); - } else { - fprintf(stderr, - "Disconnecting %s failed with status 0x%02x (%s)\n", - addr, status, mgmt_errstr(status)); - exit(EXIT_FAILURE); - } -} - -static void cmd_disconnect(int mgmt_sk, uint16_t index, int argc, char **argv) -{ - struct mgmt_cp_disconnect cp; - - if (argc < 2) { - printf("Usage: btmgmt %s <address>\n", argv[0]); - exit(EXIT_FAILURE); - } - - str2ba(argv[1], &cp.addr.bdaddr); - - if (index == MGMT_INDEX_NONE) - index = 0; - - if (mgmt_send_cmd(mgmt_sk, MGMT_OP_DISCONNECT, index, - &cp, sizeof(cp), disconnect_rsp, NULL) < 0) { - fprintf(stderr, "Unable to send disconnect cmd\n"); - exit(EXIT_FAILURE); - } -} - -static void con_rsp(int mgmt_sk, uint16_t op, uint16_t id, uint8_t status, - void *rsp, uint16_t len, void *user_data) -{ - struct mgmt_rp_get_connections *rp = rsp; - uint16_t count, i; - - if (len < sizeof(*rp)) { - fprintf(stderr, "Too small (%u bytes) get_connections rsp\n", - len); - exit(EXIT_FAILURE); - } - - count = bt_get_le16(&rp->conn_count); - if (len != sizeof(*rp) + count * sizeof(struct mgmt_addr_info)) { - fprintf(stderr, "Invalid get_connections length " - " (count=%u, len=%u)\n", count, len); - exit(EXIT_FAILURE); - } - - for (i = 0; i < count; i++) { - char addr[18]; - - ba2str(&rp->addr[i].bdaddr, addr); - - printf("%s type %s\n", addr, typestr(rp->addr[i].type)); - } - - exit(EXIT_SUCCESS); -} - -static void cmd_con(int mgmt_sk, uint16_t index, int argc, char **argv) -{ - if (index == MGMT_INDEX_NONE) - index = 0; - - if (mgmt_send_cmd(mgmt_sk, MGMT_OP_GET_CONNECTIONS, index, NULL, 0, - con_rsp, NULL) < 0) { - fprintf(stderr, "Unable to send get_connections cmd\n"); - exit(EXIT_FAILURE); - } -} - -static void find_rsp(int mgmt_sk, uint16_t op, uint16_t id, uint8_t status, - void *rsp, uint16_t len, void *user_data) -{ - if (status != 0) { - fprintf(stderr, - "Unable to start discovery. status 0x%02x (%s)\n", - status, mgmt_errstr(status)); - exit(EXIT_FAILURE); - } - - printf("Discovery started\n"); - discovery = true; -} - -static void find_usage(void) -{ - printf("Usage: btmgmt find [-l|-b]>\n"); -} - -static struct option find_options[] = { - { "help", 0, 0, 'h' }, - { "le-only", 1, 0, 'l' }, - { "bredr-only", 1, 0, 'b' }, - { 0, 0, 0, 0 } -}; - -static void cmd_find(int mgmt_sk, uint16_t index, int argc, char **argv) -{ - struct mgmt_cp_start_discovery cp; - uint8_t type; - int opt; - - if (index == MGMT_INDEX_NONE) - index = 0; - - type = 0; - hci_set_bit(BDADDR_BREDR, &type); - hci_set_bit(BDADDR_LE_PUBLIC, &type); - hci_set_bit(BDADDR_LE_RANDOM, &type); - - while ((opt = getopt_long(argc, argv, "+lbh", find_options, - NULL)) != -1) { - switch (opt) { - case 'l': - hci_clear_bit(BDADDR_BREDR, &type); - hci_set_bit(BDADDR_LE_PUBLIC, &type); - hci_set_bit(BDADDR_LE_RANDOM, &type); - break; - case 'b': - hci_set_bit(BDADDR_BREDR, &type); - hci_clear_bit(BDADDR_LE_PUBLIC, &type); - hci_clear_bit(BDADDR_LE_RANDOM, &type); - break; - case 'h': - default: - find_usage(); - exit(EXIT_SUCCESS); - } - } - - argc -= optind; - argv += optind; - optind = 0; - - memset(&cp, 0, sizeof(cp)); - cp.type = type; - - if (mgmt_send_cmd(mgmt_sk, MGMT_OP_START_DISCOVERY, index, - &cp, sizeof(cp), find_rsp, NULL) < 0) { - fprintf(stderr, "Unable to send start_discovery cmd\n"); - exit(EXIT_FAILURE); - } -} - -static void name_rsp(int mgmt_sk, uint16_t op, uint16_t id, uint8_t status, - void *rsp, uint16_t len, void *user_data) -{ - if (status != 0) { - fprintf(stderr, "Unable to set local name. status 0x%02x (%s)", - status, mgmt_errstr(status)); - exit(EXIT_FAILURE); - } - - exit(EXIT_SUCCESS); -} - -static void cmd_name(int mgmt_sk, uint16_t index, int argc, char **argv) -{ - struct mgmt_cp_set_local_name cp; - - if (argc < 2) { - printf("Usage: btmgmt %s <name> [shortname]\n", argv[0]); - exit(EXIT_FAILURE); - } - - if (index == MGMT_INDEX_NONE) - index = 0; - - memset(&cp, 0, sizeof(cp)); - strncpy((char *) cp.name, argv[1], HCI_MAX_NAME_LENGTH); - if (argc > 2) - strncpy((char *) cp.short_name, argv[2], - MGMT_MAX_SHORT_NAME_LENGTH); - - if (mgmt_send_cmd(mgmt_sk, MGMT_OP_SET_LOCAL_NAME, index, - &cp, sizeof(cp), name_rsp, NULL) < 0) { - fprintf(stderr, "Unable to send set_name cmd\n"); - exit(EXIT_FAILURE); - } -} - -static void pair_rsp(int mgmt_sk, uint16_t op, uint16_t id, uint8_t status, - void *rsp, uint16_t len, void *user_data) -{ - struct mgmt_rp_pair_device *rp = rsp; - char addr[18]; - - if (len == 0 && status != 0) { - fprintf(stderr, "Pairing failed with status 0x%02x (%s)\n", - status, mgmt_errstr(status)); - exit(EXIT_FAILURE); - } - - if (len != sizeof(*rp)) { - fprintf(stderr, "Unexpected pair_rsp len %u\n", len); - exit(EXIT_FAILURE); - } - - ba2str(&rp->addr.bdaddr, addr); - - if (status != 0) { - fprintf(stderr, - "Pairing with %s (%s) failed. status 0x%02x (%s)\n", - addr, typestr(rp->addr.type), status, - mgmt_errstr(status)); - exit(EXIT_FAILURE); - } - - printf("Paired with %s\n", addr); - - exit(EXIT_SUCCESS); -} - -static void pair_usage(void) -{ - printf("Usage: btmgmt pair [-c cap] [-t type] <remote address>\n"); -} - -static struct option pair_options[] = { - { "help", 0, 0, 'h' }, - { "capability", 1, 0, 'c' }, - { "type", 1, 0, 't' }, - { 0, 0, 0, 0 } -}; - -static void cmd_pair(int mgmt_sk, uint16_t index, int argc, char **argv) -{ - struct mgmt_cp_pair_device cp; - uint8_t cap = 0x01; - uint8_t type = BDADDR_BREDR; - int opt; - - while ((opt = getopt_long(argc, argv, "+c:t:h", pair_options, - NULL)) != -1) { - switch (opt) { - case 'c': - cap = strtol(optarg, NULL, 0); - break; - case 't': - type = strtol(optarg, NULL, 0); - break; - case 'h': - default: - pair_usage(); - exit(EXIT_SUCCESS); - } - } - - argc -= optind; - argv += optind; - optind = 0; - - if (argc < 1) { - pair_usage(); - exit(EXIT_FAILURE); - } - - if (index == MGMT_INDEX_NONE) - index = 0; - - memset(&cp, 0, sizeof(cp)); - str2ba(argv[0], &cp.addr.bdaddr); - cp.addr.type = type; - cp.io_cap = cap; - - if (mgmt_send_cmd(mgmt_sk, MGMT_OP_PAIR_DEVICE, index, &cp, sizeof(cp), - pair_rsp, NULL) < 0) { - fprintf(stderr, "Unable to send pair_device cmd\n"); - exit(EXIT_FAILURE); - } -} - -static void unpair_rsp(int mgmt_sk, uint16_t op, uint16_t id, uint8_t status, - void *rsp, uint16_t len, void *user_data) -{ - struct mgmt_rp_unpair_device *rp = rsp; - char addr[18]; - - if (len == 0 && status != 0) { - fprintf(stderr, "Unpair device failed. status 0x%02x (%s)\n", - status, mgmt_errstr(status)); - exit(EXIT_FAILURE); - } - - if (len != sizeof(*rp)) { - fprintf(stderr, "Unexpected unpair_device_rsp len %u\n", len); - exit(EXIT_FAILURE); - } - - ba2str(&rp->addr.bdaddr, addr); - - if (status != 0) { - fprintf(stderr, - "Unpairing %s failed. status 0x%02x (%s)\n", - addr, status, mgmt_errstr(status)); - exit(EXIT_FAILURE); - } - - printf("%s unpaired\n", addr); - - exit(EXIT_SUCCESS); -} - -static void cmd_unpair(int mgmt_sk, uint16_t index, int argc, char **argv) -{ - struct mgmt_cp_unpair_device cp; - - if (argc < 2) { - printf("Usage: btmgmt %s <remote address>\n", argv[0]); - exit(EXIT_FAILURE); - } - - if (index == MGMT_INDEX_NONE) - index = 0; - - memset(&cp, 0, sizeof(cp)); - str2ba(argv[1], &cp.addr.bdaddr); - cp.disconnect = 1; - - if (mgmt_send_cmd(mgmt_sk, MGMT_OP_UNPAIR_DEVICE, index, &cp, - sizeof(cp), unpair_rsp, NULL) < 0) { - fprintf(stderr, "Unable to send unpair_device cmd\n"); - exit(EXIT_FAILURE); - } -} - -static void keys_rsp(int mgmt_sk, uint16_t op, uint16_t id, uint8_t status, - void *rsp, uint16_t len, void *user_data) -{ - if (status != 0) { - fprintf(stderr, "Load keys failed with status 0x%02x (%s)\n", - status, mgmt_errstr(status)); - exit(EXIT_FAILURE); - } - - printf("Keys successfully loaded\n"); - - exit(EXIT_SUCCESS); -} - -static void cmd_keys(int mgmt_sk, uint16_t index, int argc, char **argv) -{ - struct mgmt_cp_load_link_keys cp; - - if (index == MGMT_INDEX_NONE) - index = 0; - - memset(&cp, 0, sizeof(cp)); - - if (mgmt_send_cmd(mgmt_sk, MGMT_OP_LOAD_LINK_KEYS, index, - &cp, sizeof(cp), keys_rsp, NULL) < 0) { - fprintf(stderr, "Unable to send load_keys cmd\n"); - exit(EXIT_FAILURE); - } -} - -static void block_rsp(int mgmt_sk, uint16_t op, uint16_t id, uint8_t status, - void *rsp, uint16_t len, void *user_data) -{ - struct mgmt_addr_info *rp = rsp; - char addr[18]; - - if (len == 0 && status != 0) { - fprintf(stderr, "%s failed, status 0x%02x (%s)\n", - mgmt_opstr(op), status, mgmt_errstr(status)); - exit(EXIT_FAILURE); - } - - if (len != sizeof(*rp)) { - fprintf(stderr, "Unexpected %s len %u\n", mgmt_opstr(op), len); - exit(EXIT_FAILURE); - } - - ba2str(&rp->bdaddr, addr); - - if (status != 0) { - fprintf(stderr, "%s %s (%s) failed. status 0x%02x (%s)\n", - mgmt_opstr(op), addr, typestr(rp->type), - status, mgmt_errstr(status)); - exit(EXIT_FAILURE); - } - - printf("%s %s succeeded\n", mgmt_opstr(op), addr); - - exit(EXIT_SUCCESS); -} - -static void block_usage(void) -{ - printf("Usage: btmgmt block [-t type] <remote address>\n"); -} - -static struct option block_options[] = { - { "help", 0, 0, 'h' }, - { "type", 1, 0, 't' }, - { 0, 0, 0, 0 } -}; - -static void cmd_block(int mgmt_sk, uint16_t index, int argc, char **argv) -{ - struct mgmt_cp_block_device cp; - uint8_t type = BDADDR_BREDR; - int opt; - - while ((opt = getopt_long(argc, argv, "+t:h", block_options, - NULL)) != -1) { - switch (opt) { - case 't': - type = strtol(optarg, NULL, 0); - break; - case 'h': - default: - block_usage(); - exit(EXIT_SUCCESS); - } - } - - argc -= optind; - argv += optind; - optind = 0; - - if (argc < 1) { - block_usage(); - exit(EXIT_FAILURE); - } - - if (index == MGMT_INDEX_NONE) - index = 0; - - memset(&cp, 0, sizeof(cp)); - str2ba(argv[0], &cp.addr.bdaddr); - cp.addr.type = type; - - if (mgmt_send_cmd(mgmt_sk, MGMT_OP_BLOCK_DEVICE, index, - &cp, sizeof(cp), block_rsp, NULL) < 0) { - fprintf(stderr, "Unable to send block_device cmd\n"); - exit(EXIT_FAILURE); - } -} - -static void unblock_usage(void) -{ - printf("Usage: btmgmt unblock [-t type] <remote address>\n"); -} - -static void cmd_unblock(int mgmt_sk, uint16_t index, int argc, char **argv) -{ - struct mgmt_cp_unblock_device cp; - uint8_t type = BDADDR_BREDR; - int opt; - - while ((opt = getopt_long(argc, argv, "+t:h", block_options, - NULL)) != -1) { - switch (opt) { - case 't': - type = strtol(optarg, NULL, 0); - break; - case 'h': - default: - unblock_usage(); - exit(EXIT_SUCCESS); - } - } - - argc -= optind; - argv += optind; - optind = 0; - - if (argc < 1) { - unblock_usage(); - exit(EXIT_FAILURE); - } - - if (index == MGMT_INDEX_NONE) - index = 0; - - memset(&cp, 0, sizeof(cp)); - str2ba(argv[0], &cp.addr.bdaddr); - cp.addr.type = type; - - if (mgmt_send_cmd(mgmt_sk, MGMT_OP_UNBLOCK_DEVICE, index, - &cp, sizeof(cp), block_rsp, NULL) < 0) { - fprintf(stderr, "Unable to send unblock_device cmd\n"); - exit(EXIT_FAILURE); - } -} - -static void uuid_to_uuid128(uuid_t *uuid128, const uuid_t *uuid) -{ - if (uuid->type == SDP_UUID16) - sdp_uuid16_to_uuid128(uuid128, uuid); - else if (uuid->type == SDP_UUID32) - sdp_uuid32_to_uuid128(uuid128, uuid); - else - memcpy(uuid128, uuid, sizeof(*uuid)); -} - -static void cmd_add_uuid(int mgmt_sk, uint16_t index, int argc, char **argv) -{ - struct mgmt_cp_add_uuid cp; - uint128_t uint128; - uuid_t uuid, uuid128; - - if (argc < 3) { - printf("UUID and service hint needed\n"); - exit(EXIT_FAILURE); - } - - if (index == MGMT_INDEX_NONE) - index = 0; - - if (bt_string2uuid(&uuid, argv[1]) < 0) { - printf("Invalid UUID: %s\n", argv[1]); - exit(EXIT_FAILURE); - } - - memset(&cp, 0, sizeof(cp)); - - uuid_to_uuid128(&uuid128, &uuid); - ntoh128((uint128_t *) uuid128.value.uuid128.data, &uint128); - htob128(&uint128, (uint128_t *) cp.uuid); - - cp.svc_hint = atoi(argv[2]); - - if (mgmt_send_cmd(mgmt_sk, MGMT_OP_ADD_UUID, index, - &cp, sizeof(cp), class_rsp, NULL) < 0) { - fprintf(stderr, "Unable to send add_uuid cmd\n"); - exit(EXIT_FAILURE); - } -} - -static void cmd_remove_uuid(int mgmt_sk, uint16_t index, int argc, char **argv) -{ - struct mgmt_cp_remove_uuid cp; - uint128_t uint128; - uuid_t uuid, uuid128; - - if (argc < 2) { - printf("UUID needed\n"); - exit(EXIT_FAILURE); - } - - if (index == MGMT_INDEX_NONE) - index = 0; - - if (bt_string2uuid(&uuid, argv[1]) < 0) { - printf("Invalid UUID: %s\n", argv[1]); - exit(EXIT_FAILURE); - } - - memset(&cp, 0, sizeof(cp)); - - uuid_to_uuid128(&uuid128, &uuid); - ntoh128((uint128_t *) uuid128.value.uuid128.data, &uint128); - htob128(&uint128, (uint128_t *) cp.uuid); - - if (mgmt_send_cmd(mgmt_sk, MGMT_OP_REMOVE_UUID, index, - &cp, sizeof(cp), class_rsp, NULL) < 0) { - fprintf(stderr, "Unable to send remove_uuid cmd\n"); - exit(EXIT_FAILURE); - } -} - -static void cmd_clr_uuids(int mgmt_sk, uint16_t index, int argc, char **argv) -{ - char *uuid_any = "00000000-0000-0000-0000-000000000000"; - char *rm_argv[] = { "rm-uuid", uuid_any, NULL }; - - cmd_remove_uuid(mgmt_sk, index, 2, rm_argv); -} - -static void did_rsp(int mgmt_sk, uint16_t op, uint16_t id, uint8_t status, - void *rsp, uint16_t len, void *user_data) -{ - if (status != 0) { - fprintf(stderr, "Set Device ID failed with status 0x%02x (%s)\n", - status, mgmt_errstr(status)); - exit(EXIT_FAILURE); - } - - printf("Device ID successfully set\n"); - - exit(EXIT_SUCCESS); -} - -static void did_usage(void) -{ - printf("Usage: btmgmt did <source>:<vendor>:<product>:<version>\n"); - printf(" possible source values: bluetooth, usb\n"); -} - -static void cmd_did(int mgmt_sk, uint16_t index, int argc, char **argv) -{ - struct mgmt_cp_set_device_id cp; - uint16_t vendor, product, version , source; - int result; - - if (argc < 2) { - did_usage(); - exit(EXIT_FAILURE); - } - - result = sscanf(argv[1], "bluetooth:%4hx:%4hx:%4hx", &vendor, &product, - &version); - if (result == 3) { - source = 0x0001; - goto done; - } - - result = sscanf(argv[1], "usb:%4hx:%4hx:%4hx", &vendor, &product, - &version); - if (result == 3) { - source = 0x0002; - goto done; - } - - did_usage(); - exit(EXIT_FAILURE); - -done: - if (index == MGMT_INDEX_NONE) - index = 0; - - cp.source = htobs(source); - cp.vendor = htobs(vendor); - cp.product = htobs(product); - cp.version = htobs(version); - - if (mgmt_send_cmd(mgmt_sk, MGMT_OP_SET_DEVICE_ID, index, - &cp, sizeof(cp), did_rsp, NULL) < 0) { - fprintf(stderr, "Unable to send set_dev_class cmd\n"); - exit(EXIT_FAILURE); - } -} - -static struct { - char *cmd; - void (*func)(int mgmt_sk, uint16_t index, int argc, char **argv); - char *doc; -} command[] = { - { "monitor", cmd_monitor, "Monitor events" }, - { "version", cmd_version, "Get the MGMT Version" }, - { "commands", cmd_commands, "List supported commands" }, - { "info", cmd_info, "Show controller info" }, - { "power", cmd_power, "Toggle powered state" }, - { "discov", cmd_discov, "Toggle discoverable state" }, - { "connectable",cmd_connectable,"Toggle connectable state" }, - { "pairable", cmd_pairable, "Toggle pairable state" }, - { "linksec", cmd_linksec, "Toggle link level security" }, - { "ssp", cmd_ssp, "Toggle SSP mode" }, - { "hs", cmd_hs, "Toggle HS Support" }, - { "le", cmd_le, "Toggle LE Support" }, - { "class", cmd_class, "Set device major/minor class" }, - { "disconnect", cmd_disconnect, "Disconnect device" }, - { "con", cmd_con, "List connections" }, - { "find", cmd_find, "Discover nearby devices" }, - { "name", cmd_name, "Set local name" }, - { "pair", cmd_pair, "Pair with a remote device" }, - { "unpair", cmd_unpair, "Unpair device" }, - { "keys", cmd_keys, "Load Keys" }, - { "block", cmd_block, "Block Device" }, - { "unblock", cmd_unblock, "Unblock Device" }, - { "add-uuid", cmd_add_uuid, "Add UUID" }, - { "rm-uuid", cmd_add_uuid, "Remove UUID" }, - { "clr-uuids", cmd_clr_uuids, "Clear UUIDs", }, - { "did", cmd_did, "Set Device ID", }, - { NULL, NULL, 0 } -}; - -static void usage(void) -{ - int i; - - printf("btmgmt ver %s\n", VERSION); - printf("Usage:\n" - "\tbtmgmt [options] <command> [command parameters]\n"); - - printf("Options:\n" - "\t--index <id>\tSpecify adapter index\n" - "\t--verbose\tEnable extra logging\n" - "\t--help\tDisplay help\n"); - - printf("Commands:\n"); - for (i = 0; command[i].cmd; i++) - printf("\t%-15s\t%s\n", command[i].cmd, command[i].doc); - - printf("\n" - "For more information on the usage of each command use:\n" - "\tbtmgmt <command> --help\n" ); -} - -static struct option main_options[] = { - { "index", 1, 0, 'i' }, - { "verbose", 0, 0, 'v' }, - { "help", 0, 0, 'h' }, - { 0, 0, 0, 0 } -}; - -int main(int argc, char *argv[]) -{ - int opt, i, mgmt_sk; - uint16_t index = MGMT_INDEX_NONE; - struct pollfd pollfd; - - while ((opt = getopt_long(argc, argv, "+hvi:", - main_options, NULL)) != -1) { - switch (opt) { - case 'i': - if (strlen(optarg) > 3 && - strncasecmp(optarg, "hci", 3) == 0) - index = atoi(&optarg[4]); - else - index = atoi(optarg); - break; - case 'v': - monitor = true; - break; - case 'h': - default: - usage(); - return 0; - } - } - - argc -= optind; - argv += optind; - optind = 0; - - if (argc < 1) { - usage(); - return 0; - } - - mgmt_sk = mgmt_open(); - if (mgmt_sk < 0) { - fprintf(stderr, "Unable to open mgmt socket\n"); - return -1; - } - - for (i = 0; command[i].cmd; i++) { - if (strcmp(command[i].cmd, argv[0]) != 0) - continue; - - command[i].func(mgmt_sk, index, argc, argv); - break; - } - - if (command[i].cmd == NULL) { - fprintf(stderr, "Unknown command: %s\n", argv[0]); - close(mgmt_sk); - return -1; - } - - pollfd.fd = mgmt_sk; - pollfd.events = POLLIN; - pollfd.revents = 0; - - while (poll(&pollfd, 1, -1) >= 0) { - if (pollfd.revents & (POLLHUP | POLLERR | POLLNVAL)) - break; - - if (pollfd.revents & POLLIN) - mgmt_process_data(mgmt_sk); - - pollfd.revents = 0; - } - - close(mgmt_sk); - - return 0; -} diff --git a/tools/mgmt/main.c b/tools/mgmt/main.c new file mode 100644 index 0000000..b2d6c3c --- /dev/null +++ b/tools/mgmt/main.c @@ -0,0 +1,1933 @@ +/* + * BlueZ - Bluetooth protocol stack for Linux + * + * Copyright (C) 2011 Intel Corporation. All rights reserved. + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA + * + */ + +#ifdef HAVE_CONFIG_H +#include <config.h> +#endif + +#include <stdlib.h> +#include <stdio.h> +#include <unistd.h> +#include <errno.h> +#include <string.h> +#include <sys/types.h> +#include <sys/socket.h> +#include <poll.h> +#include <getopt.h> +#include <stdbool.h> + +#include <bluetooth/bluetooth.h> +#include <bluetooth/hci.h> +#include <bluetooth/hci_lib.h> +#include <bluetooth/sdp.h> +#include <bluetooth/sdp_lib.h> +#include <bluetooth/mgmt.h> + +#include <glib.h> +#include "glib-helper.h" + +static bool monitor = false; +static bool discovery = false; +static bool resolve_names = true; + +typedef void (*cmd_cb)(int mgmt_sk, uint16_t op, uint16_t id, uint8_t status, + void *rsp, uint16_t len, void *user_data); + +static struct pending_cmd { + uint16_t op; + uint16_t id; + cmd_cb cb; + void *user_data; + struct pending_cmd *next; +} *pending = NULL; + +static int mgmt_send_cmd(int mgmt_sk, uint16_t op, uint16_t id, void *data, + size_t len, cmd_cb func, void *user_data) +{ + char buf[1024]; + struct pending_cmd *cmd; + struct mgmt_hdr *hdr = (void *) buf; + + if (len + MGMT_HDR_SIZE > sizeof(buf)) + return -EINVAL; + + cmd = calloc(1, sizeof(struct pending_cmd)); + if (cmd == NULL) + return -errno; + + cmd->op = op; + cmd->id = id; + cmd->cb = func; + cmd->user_data = user_data; + + memset(buf, 0, sizeof(buf)); + hdr->opcode = htobs(op); + hdr->index = htobs(id); + hdr->len = htobs(len); + memcpy(buf + MGMT_HDR_SIZE, data, len); + + if (write(mgmt_sk, buf, MGMT_HDR_SIZE + len) < 0) { + fprintf(stderr, "Unable to write to socket: %s\n", + strerror(errno)); + free(cmd); + return -1; + } + + cmd->next = pending; + pending = cmd; + + return 0; +} + +static int mgmt_open(void) +{ + struct sockaddr_hci addr; + int sk; + + sk = socket(AF_BLUETOOTH, SOCK_RAW, BTPROTO_HCI); + if (sk < 0) { + fprintf(stderr, "socket: %s\n", strerror(errno)); + return sk; + } + + memset(&addr, 0, sizeof(addr)); + addr.hci_family = AF_BLUETOOTH; + addr.hci_dev = HCI_DEV_NONE; + addr.hci_channel = HCI_CHANNEL_CONTROL; + + if (bind(sk, (struct sockaddr *) &addr, sizeof(addr)) < 0) { + fprintf(stderr, "bind: %s\n", strerror(errno)); + close(sk); + return -1; + } + + return sk; +} + +static void mgmt_check_pending(int mgmt_sk, uint16_t op, uint16_t index, + uint16_t status, void *data, uint16_t len) +{ + struct pending_cmd *c, *prev; + + for (c = pending, prev = NULL; c != NULL; prev = c, c = c->next) { + if (c->op != op) + continue; + if (c->id != index) + continue; + + if (c == pending) + pending = c->next; + else + prev->next = c->next; + + c->cb(mgmt_sk, op, index, status, data, len, c->user_data); + + free(c); + break; + } +} + +static int mgmt_cmd_complete(int mgmt_sk, uint16_t index, + struct mgmt_ev_cmd_complete *ev, uint16_t len) +{ + uint16_t op; + + if (len < sizeof(*ev)) { + fprintf(stderr, "Too short (%u bytes) cmd complete event\n", + len); + return -EINVAL; + } + + op = bt_get_le16(&ev->opcode); + + len -= sizeof(*ev); + + if (monitor) + printf("%s complete, opcode 0x%04x len %u\n", mgmt_opstr(op), + op, len); + + mgmt_check_pending(mgmt_sk, op, index, ev->status, ev->data, len); + + return 0; +} + +static int mgmt_cmd_status(int mgmt_sk, uint16_t index, + struct mgmt_ev_cmd_status *ev, uint16_t len) +{ + uint16_t opcode; + + if (len < sizeof(*ev)) { + fprintf(stderr, "Too short (%u bytes) cmd status event\n", + len); + return -EINVAL; + } + + opcode = bt_get_le16(&ev->opcode); + + if (monitor) + printf("cmd status, opcode 0x%04x status 0x%02x (%s)\n", + opcode, ev->status, mgmt_errstr(ev->status)); + + if (ev->status != 0) + mgmt_check_pending(mgmt_sk, opcode, index, ev->status, + NULL, 0); + + return 0; +} + +static int mgmt_controller_error(uint16_t index, + struct mgmt_ev_controller_error *ev, + uint16_t len) +{ + if (len < sizeof(*ev)) { + fprintf(stderr, + "Too short (%u bytes) controller error event\n", len); + return -EINVAL; + } + + if (monitor) + printf("hci%u error 0x%02x\n", index, ev->error_code); + + return 0; +} + +static int mgmt_index_added(int mgmt_sk, uint16_t index) +{ + if (monitor) + printf("hci%u added\n", index); + return 0; +} + +static int mgmt_index_removed(int mgmt_sk, uint16_t index) +{ + if (monitor) + printf("hci%u removed\n", index); + return 0; +} + +static const char *settings_str[] = { + "powered", + "connectable", + "fast-connectable", + "discoverable", + "pairable", + "link-security", + "ssp", + "br/edr", + "hs", + "le" , +}; + +static void print_settings(uint32_t settings) +{ + unsigned i; + + for (i = 0; i < NELEM(settings_str); i++) { + if ((settings & (1 << i)) != 0) + printf("%s ", settings_str[i]); + } +} + +static int mgmt_new_settings(int mgmt_sk, uint16_t index, + uint32_t *ev, uint16_t len) +{ + if (len < sizeof(*ev)) { + fprintf(stderr, "Too short new_settings event (%u)\n", len); + return -EINVAL; + } + + if (monitor) { + printf("hci%u new_settings: ", index); + print_settings(bt_get_le32(ev)); + printf("\n"); + } + + return 0; +} + +static int mgmt_discovering(int mgmt_sk, uint16_t index, + struct mgmt_ev_discovering *ev, uint16_t len) +{ + if (len < sizeof(*ev)) { + fprintf(stderr, "Too short (%u bytes) discovering event\n", + len); + return -EINVAL; + } + + if (ev->discovering == 0 && discovery) + exit(EXIT_SUCCESS); + + if (monitor) + printf("hci%u type %u discovering %s\n", index, + ev->type, ev->discovering ? "on" : "off"); + + return 0; +} + +static int mgmt_new_link_key(int mgmt_sk, uint16_t index, + struct mgmt_ev_new_link_key *ev, uint16_t len) +{ + + if (len != sizeof(*ev)) { + fprintf(stderr, "Invalid new_link_key length (%u bytes)\n", + len); + return -EINVAL; + } + + if (monitor) { + char addr[18]; + ba2str(&ev->key.addr.bdaddr, addr); + printf("hci%u new_link_key %s type 0x%02x pin_len %d " + "store_hint %u\n", index, addr, ev->key.type, + ev->key.pin_len, ev->store_hint); + } + + return 0; +} + +static const char *typestr(uint8_t type) +{ + const char *str[] = { "BR/EDR", "LE Public", "LE Random" }; + + if (type <= BDADDR_LE_RANDOM) + return str[type]; + + return "(unknown)"; +} + +static int mgmt_connected(int mgmt_sk, uint16_t index, + struct mgmt_ev_device_connected *ev, + uint16_t len) +{ + uint16_t eir_len; + + if (len < sizeof(*ev)) { + fprintf(stderr, + "Invalid connected event length (%u bytes)\n", len); + return -EINVAL; + } + + eir_len = bt_get_le16(&ev->eir_len); + if (len != sizeof(*ev) + eir_len) { + fprintf(stderr, "Invalid connected event length " + "(%u bytes, eir_len %u bytes)\n", len, eir_len); + return -EINVAL; + } + + if (monitor) { + char addr[18]; + ba2str(&ev->addr.bdaddr, addr); + printf("hci%u %s type %s connected eir_len %u\n", index, addr, + typestr(ev->addr.type), eir_len); + } + + return 0; +} + +static int mgmt_disconnected(int mgmt_sk, uint16_t index, + struct mgmt_addr_info *ev, uint16_t len) +{ + if (len != sizeof(*ev)) { + fprintf(stderr, + "Invalid disconnected event length (%u bytes)\n", len); + return -EINVAL; + } + + if (monitor) { + char addr[18]; + ba2str(&ev->bdaddr, addr); + printf("hci%u %s type %s disconnected\n", index, addr, + typestr(ev->type)); + } + + return 0; +} + +static int mgmt_conn_failed(int mgmt_sk, uint16_t index, + struct mgmt_ev_connect_failed *ev, + uint16_t len) +{ + if (len != sizeof(*ev)) { + fprintf(stderr, + "Invalid connect_failed event length (%u bytes)\n", len); + return -EINVAL; + } + + if (monitor) { + char addr[18]; + ba2str(&ev->addr.bdaddr, addr); + printf("hci%u %s type %s connect failed (status 0x%02x, %s)\n", + index, addr, typestr(ev->addr.type), ev->status, + mgmt_errstr(ev->status)); + } + + return 0; +} + +static int mgmt_auth_failed(int mgmt_sk, uint16_t index, + struct mgmt_ev_auth_failed *ev, + uint16_t len) +{ + if (len != sizeof(*ev)) { + fprintf(stderr, + "Invalid auth_failed event length (%u bytes)\n", len); + return -EINVAL; + } + + if (monitor) { + char addr[18]; + ba2str(&ev->addr.bdaddr, addr); + printf("hci%u %s auth failed with status 0x%02x (%s)\n", + index, addr, ev->status, mgmt_errstr(ev->status)); + } + + return 0; +} + +static int mgmt_name_changed(int mgmt_sk, uint16_t index, + struct mgmt_ev_local_name_changed *ev, + uint16_t len) +{ + if (len != sizeof(*ev)) { + fprintf(stderr, + "Invalid local_name_changed length (%u bytes)\n", len); + return -EINVAL; + } + + if (monitor) + printf("hci%u name changed: %s\n", index, ev->name); + + return 0; +} + +static void confirm_name_rsp(int mgmt_sk, uint16_t op, uint16_t id, + uint8_t status, void *rsp, uint16_t len, + void *user_data) +{ + struct mgmt_rp_confirm_name *rp = rsp; + char addr[18]; + + if (len == 0 && status != 0) { + fprintf(stderr, + "hci%u confirm_name failed with status 0x%02x (%s)\n", + id, status, mgmt_errstr(status)); + return; + } + + if (len != sizeof(*rp)) { + fprintf(stderr, + "hci%u confirm_name rsp length %u instead of %zu\n", + id, len, sizeof(*rp)); + return; + } + + ba2str(&rp->addr.bdaddr, addr); + + if (status != 0) + fprintf(stderr, + "hci%u confirm_name for %s failed: 0x%02x (%s)\n", + id, addr, status, mgmt_errstr(status)); + else + printf("hci%u confirm_name succeeded for %s\n", id, addr); +} + +static int mgmt_device_found(int mgmt_sk, uint16_t index, + struct mgmt_ev_device_found *ev, uint16_t len) +{ + uint32_t flags; + uint16_t eir_len; + + if (len < sizeof(*ev)) { + fprintf(stderr, + "Too short device_found length (%u bytes)\n", len); + return -EINVAL; + } + + flags = btohs(ev->flags); + + eir_len = bt_get_le16(&ev->eir_len); + if (len != sizeof(*ev) + eir_len) { + fprintf(stderr, "dev_found: expected %zu bytes, got %u bytes", + sizeof(*ev) + eir_len, len); + return -EINVAL; + } + + if (monitor || discovery) { + char addr[18]; + ba2str(&ev->addr.bdaddr, addr); + printf("hci%u dev_found: %s type %s rssi %d " + "flags 0x%04x eir_len %u\n", index, addr, + typestr(ev->addr.type), ev->rssi, flags, eir_len); + } + + if (discovery && (flags & MGMT_DEV_FOUND_CONFIRM_NAME)) { + struct mgmt_cp_confirm_name cp; + + memset(&cp, 0, sizeof(cp)); + memcpy(&cp.addr, &ev->addr, sizeof(cp.addr)); + if (resolve_names) + cp.name_known = 0; + else + cp.name_known = 1; + + mgmt_send_cmd(mgmt_sk, MGMT_OP_CONFIRM_NAME, index, + &cp, sizeof(cp), confirm_name_rsp, + NULL); + } + + return 0; +} + +static void pin_rsp(int mgmt_sk, uint16_t op, uint16_t id, uint8_t status, + void *rsp, uint16_t len, void *user_data) +{ + if (status != 0) { + fprintf(stderr, + "hci%u PIN Code reply failed with status 0x%02x (%s)", + id, status, mgmt_errstr(status)); + exit(EXIT_FAILURE); + } + + printf("hci%u PIN Reply successful\n", id); +} + +static int mgmt_pin_reply(int mgmt_sk, uint16_t index, + struct mgmt_addr_info *addr, + const char *pin, size_t len) +{ + struct mgmt_cp_pin_code_reply cp; + + memset(&cp, 0, sizeof(cp)); + memcpy(&cp.addr, addr, sizeof(cp.addr)); + cp.pin_len = len; + memcpy(cp.pin_code, pin, len); + + return mgmt_send_cmd(mgmt_sk, MGMT_OP_PIN_CODE_REPLY, index, + &cp, sizeof(cp), pin_rsp, NULL); +} + +static void pin_neg_rsp(int mgmt_sk, uint16_t op, uint16_t id, uint8_t status, + void *rsp, uint16_t len, void *user_data) +{ + if (status != 0) { + fprintf(stderr, + "hci%u PIN Neg reply failed with status 0x%02x (%s)", + id, status, mgmt_errstr(status)); + exit(EXIT_FAILURE); + } + + printf("hci%u PIN Negative Reply successful\n", id); +} + +static int mgmt_pin_neg_reply(int mgmt_sk, uint16_t index, + struct mgmt_addr_info *addr) +{ + struct mgmt_cp_pin_code_neg_reply cp; + + memset(&cp, 0, sizeof(cp)); + memcpy(&cp.addr, addr, sizeof(cp.addr)); + + return mgmt_send_cmd(mgmt_sk, MGMT_OP_PIN_CODE_NEG_REPLY, index, + &cp, sizeof(cp), pin_neg_rsp, NULL); +} + +static int mgmt_request_pin(int mgmt_sk, uint16_t index, + struct mgmt_ev_pin_code_request *ev, + uint16_t len) +{ + char pin[18]; + size_t pin_len; + + if (len != sizeof(*ev)) { + fprintf(stderr, + "Invalid pin_code request length (%u bytes)\n", len); + return -EINVAL; + } + + if (monitor) { + char addr[18]; + ba2str(&ev->addr.bdaddr, addr); + printf("hci%u %s request PIN\n", index, addr); + } + + printf("PIN Request (press enter to reject) >> "); + fflush(stdout); + + memset(pin, 0, sizeof(pin)); + + if (fgets(pin, sizeof(pin), stdin) == NULL || pin[0] == '\n') + return mgmt_pin_neg_reply(mgmt_sk, index, &ev->addr); + + pin_len = strlen(pin); + if (pin[pin_len - 1] == '\n') { + pin[pin_len - 1] = '\0'; + pin_len--; + } + + return mgmt_pin_reply(mgmt_sk, index, &ev->addr, pin, pin_len); +} + +static void confirm_rsp(int mgmt_sk, uint16_t op, uint16_t id, uint8_t status, + void *rsp, uint16_t len, void *user_data) +{ + if (status != 0) { + fprintf(stderr, + "hci%u User Confirm reply failed. status 0x%02x (%s)", + id, status, mgmt_errstr(status)); + exit(EXIT_FAILURE); + } + + printf("hci%u User Confirm Reply successful\n", id); +} + +static int mgmt_confirm_reply(int mgmt_sk, uint16_t index, bdaddr_t *bdaddr) +{ + struct mgmt_cp_user_confirm_reply cp; + + memset(&cp, 0, sizeof(cp)); + bacpy(&cp.addr.bdaddr, bdaddr); + + return mgmt_send_cmd(mgmt_sk, MGMT_OP_USER_CONFIRM_REPLY, index, + &cp, sizeof(cp), confirm_rsp, NULL); +} + +static void confirm_neg_rsp(int mgmt_sk, uint16_t op, uint16_t id, + uint8_t status, void *rsp, uint16_t len, + void *user_data) +{ + if (status != 0) { + fprintf(stderr, + "hci%u Confirm Neg reply failed. status 0x%02x (%s)", + id, status, mgmt_errstr(status)); + exit(EXIT_FAILURE); + } + + printf("hci%u User Confirm Negative Reply successful\n", id); +} + +static int mgmt_confirm_neg_reply(int mgmt_sk, uint16_t index, + bdaddr_t *bdaddr) +{ + struct mgmt_cp_user_confirm_reply cp; + + memset(&cp, 0, sizeof(cp)); + bacpy(&cp.addr.bdaddr, bdaddr); + + return mgmt_send_cmd(mgmt_sk, MGMT_OP_USER_CONFIRM_NEG_REPLY, index, + &cp, sizeof(cp), confirm_neg_rsp, NULL); +} + + +static int mgmt_user_confirm(int mgmt_sk, uint16_t index, + struct mgmt_ev_user_confirm_request *ev, + uint16_t len) +{ + char rsp[5]; + size_t rsp_len; + uint32_t val; + char addr[18]; + + if (len != sizeof(*ev)) { + fprintf(stderr, + "Invalid user_confirm request length (%u)\n", len); + return -EINVAL; + } + + ba2str(&ev->addr.bdaddr, addr); + val = bt_get_le32(&ev->value); + + if (monitor) + printf("hci%u %s User Confirm %06u hint %u\n", index, addr, + val, ev->confirm_hint); + + if (ev->confirm_hint) + printf("Accept pairing with %s (yes/no) >> ", addr); + else + printf("Confirm value %06u for %s (yes/no) >> ", val, addr); + + fflush(stdout); + + memset(rsp, 0, sizeof(rsp)); + + if (fgets(rsp, sizeof(rsp), stdin) == NULL || rsp[0] == '\n') + return mgmt_confirm_neg_reply(mgmt_sk, index, &ev->addr.bdaddr); + + rsp_len = strlen(rsp); + if (rsp[rsp_len - 1] == '\n') { + rsp[rsp_len - 1] = '\0'; + rsp_len--; + } + + if (rsp[0] == 'y' || rsp[0] == 'Y') + return mgmt_confirm_reply(mgmt_sk, index, &ev->addr.bdaddr); + else + return mgmt_confirm_neg_reply(mgmt_sk, index, &ev->addr.bdaddr); +} + +static int mgmt_handle_event(int mgmt_sk, uint16_t ev, uint16_t index, + void *data, uint16_t len) +{ + if (monitor) + printf("event: %s\n", mgmt_evstr(ev)); + + switch (ev) { + case MGMT_EV_CMD_COMPLETE: + return mgmt_cmd_complete(mgmt_sk, index, data, len); + case MGMT_EV_CMD_STATUS: + return mgmt_cmd_status(mgmt_sk, index, data, len); + case MGMT_EV_CONTROLLER_ERROR: + return mgmt_controller_error(index, data, len); + case MGMT_EV_INDEX_ADDED: + return mgmt_index_added(mgmt_sk, index); + case MGMT_EV_INDEX_REMOVED: + return mgmt_index_removed(mgmt_sk, index); + case MGMT_EV_NEW_SETTINGS: + return mgmt_new_settings(mgmt_sk, index, data, len); + case MGMT_EV_DISCOVERING: + return mgmt_discovering(mgmt_sk, index, data, len); + case MGMT_EV_NEW_LINK_KEY: + return mgmt_new_link_key(mgmt_sk, index, data, len); + case MGMT_EV_DEVICE_CONNECTED: + return mgmt_connected(mgmt_sk, index, data, len); + case MGMT_EV_DEVICE_DISCONNECTED: + return mgmt_disconnected(mgmt_sk, index, data, len); + case MGMT_EV_CONNECT_FAILED: + return mgmt_conn_failed(mgmt_sk, index, data, len); + case MGMT_EV_AUTH_FAILED: + return mgmt_auth_failed(mgmt_sk, index, data, len); + case MGMT_EV_LOCAL_NAME_CHANGED: + return mgmt_name_changed(mgmt_sk, index, data, len); + case MGMT_EV_DEVICE_FOUND: + return mgmt_device_found(mgmt_sk, index, data, len); + case MGMT_EV_PIN_CODE_REQUEST: + return mgmt_request_pin(mgmt_sk, index, data, len); + case MGMT_EV_USER_CONFIRM_REQUEST: + return mgmt_user_confirm(mgmt_sk, index, data, len); + default: + if (monitor) + printf("Unhandled event 0x%04x (%s)\n", ev, mgmt_evstr(ev)); + return 0; + } +} + +static int mgmt_process_data(int mgmt_sk) +{ + char buf[1024]; + struct mgmt_hdr *hdr = (void *) buf; + uint16_t len, ev, index; + ssize_t ret; + + ret = read(mgmt_sk, buf, sizeof(buf)); + if (ret < 0) { + fprintf(stderr, "read: %s\n", strerror(errno)); + return ret; + } + + if (ret < MGMT_HDR_SIZE) { + fprintf(stderr, "Too small mgmt packet (%zd bytes)\n", ret); + return 0; + } + + ev = bt_get_le16(&hdr->opcode); + index = bt_get_le16(&hdr->index); + len = bt_get_le16(&hdr->len); + + if (monitor) + printf("event 0x%04x len 0x%04x index 0x%04x\n", ev, len, index); + + if (ret != MGMT_HDR_SIZE + len) { + fprintf(stderr, "Packet length mismatch. ret %zd len %u", + ret, len); + return 0; + } + + mgmt_handle_event(mgmt_sk, ev, index, buf + MGMT_HDR_SIZE, len); + + return 0; +} + +static void cmd_monitor(int mgmt_sk, uint16_t index, int argc, char **argv) +{ + printf("Monitoring mgmt events...\n"); + monitor = true; +} + +static void version_rsp(int mgmt_sk, uint16_t op, uint16_t id, uint8_t status, + void *rsp, uint16_t len, void *user_data) +{ + struct mgmt_rp_read_version *rp = rsp; + + if (status != 0) { + fprintf(stderr, "Reading mgmt version failed with status" + " 0x%02x (%s)\n", status, mgmt_errstr(status)); + exit(EXIT_FAILURE); + } + + if (len < sizeof(*rp)) { + fprintf(stderr, "Too small version reply (%u bytes)\n", len); + exit(EXIT_FAILURE); + } + + printf("MGMT Version %u, revision %u\n", rp->version, + bt_get_le16(&rp->revision)); + + exit(EXIT_SUCCESS); +} + +static void cmd_version(int mgmt_sk, uint16_t index, int argc, char **argv) +{ + if (mgmt_send_cmd(mgmt_sk, MGMT_OP_READ_VERSION, MGMT_INDEX_NONE, + NULL, 0, version_rsp, NULL) < 0) { + fprintf(stderr, "Unable to send read_version cmd\n"); + exit(EXIT_FAILURE); + } +} + +static void commands_rsp(int mgmt_sk, uint16_t op, uint16_t id, uint8_t status, + void *rsp, uint16_t len, void *user_data) +{ + struct mgmt_rp_read_commands *rp = rsp; + uint16_t num_commands, num_events, *opcode; + size_t expected_len; + int i; + + if (status != 0) { + fprintf(stderr, "Reading supported commands failed with status" + " 0x%02x (%s)\n", status, mgmt_errstr(status)); + exit(EXIT_FAILURE); + } + + if (len < sizeof(*rp)) { + fprintf(stderr, "Too small commands reply (%u bytes)\n", len); + exit(EXIT_FAILURE); + } + + num_commands = bt_get_le16(&rp->num_commands); + num_events = bt_get_le16(&rp->num_events); + + expected_len = sizeof(*rp) + num_commands * sizeof(uint16_t) + + num_events * sizeof(uint16_t); + + if (len < expected_len) { + fprintf(stderr, "Too small commands reply (%u != %zu)\n", + len, expected_len); + exit(EXIT_FAILURE); + } + + opcode = rp->opcodes; + + printf("%u commands:\n", num_commands); + for (i = 0; i < num_commands; i++) { + uint16_t op = bt_get_le16(opcode++); + printf("\t%s (0x%04x)\n", mgmt_opstr(op), op); + } + + printf("%u events:\n", num_events); + for (i = 0; i < num_events; i++) { + uint16_t ev = bt_get_le16(opcode++); + printf("\t%s (0x%04x)\n", mgmt_evstr(ev), ev); + } + + exit(EXIT_SUCCESS); +} + +static void cmd_commands(int mgmt_sk, uint16_t index, int argc, char **argv) +{ + if (mgmt_send_cmd(mgmt_sk, MGMT_OP_READ_COMMANDS, MGMT_INDEX_NONE, + NULL, 0, commands_rsp, NULL) < 0) { + fprintf(stderr, "Unable to send read_commands cmd\n"); + exit(EXIT_FAILURE); + } +} + +static void info_rsp(int mgmt_sk, uint16_t op, uint16_t id, uint8_t status, + void *rsp, uint16_t len, void *user_data) +{ + struct mgmt_rp_read_info *rp = rsp; + char addr[18]; + + if (status != 0) { + fprintf(stderr, + "Reading hci%u info failed with status 0x%02x (%s)\n", + id, status, mgmt_errstr(status)); + exit(EXIT_FAILURE); + } + + if (len < sizeof(*rp)) { + fprintf(stderr, "Too small info reply (%u bytes)\n", len); + exit(EXIT_FAILURE); + } + + ba2str(&rp->bdaddr, addr); + printf("hci%u:\taddr %s version %u manufacturer %u" + " class 0x%02x%02x%02x\n", + id, addr, rp->version, bt_get_le16(&rp->manufacturer), + rp->dev_class[2], rp->dev_class[1], rp->dev_class[0]); + + printf("\tsupported settings: "); + print_settings(bt_get_le32(&rp->supported_settings)); + + printf("\n\tcurrent settings: "); + print_settings(bt_get_le32(&rp->current_settings)); + + printf("\n\tname %s\n", rp->name); + printf("\tshort name %s\n", rp->short_name); + + if (pending == NULL) + exit(EXIT_SUCCESS); +} + +static void index_rsp(int mgmt_sk, uint16_t op, uint16_t id, uint8_t status, + void *rsp, uint16_t len, void *user_data) +{ + struct mgmt_rp_read_index_list *rp = rsp; + uint16_t count; + unsigned int i; + + if (status != 0) { + fprintf(stderr, + "Reading index list failed with status 0x%02x (%s)\n", + status, mgmt_errstr(status)); + exit(EXIT_FAILURE); + } + + if (len < sizeof(*rp)) { + fprintf(stderr, "Too small index list reply (%u bytes)\n", + len); + exit(EXIT_FAILURE); + } + + count = bt_get_le16(&rp->num_controllers); + + if (len < sizeof(*rp) + count * sizeof(uint16_t)) { + fprintf(stderr, + "Index count (%u) doesn't match reply length (%u)\n", + count, len); + exit(EXIT_FAILURE); + } + + if (monitor) + printf("Index list with %u item%s\n", + count, count > 1 ? "s" : ""); + + if (count == 0) + exit(EXIT_SUCCESS); + + if (monitor && count > 0) + printf("\t"); + + for (i = 0; i < count; i++) { + uint16_t index; + + index = bt_get_le16(&rp->index[i]); + + if (monitor) + printf("hci%u ", index); + + if (mgmt_send_cmd(mgmt_sk, MGMT_OP_READ_INFO, index, NULL, + 0, info_rsp, NULL) < 0) { + fprintf(stderr, "Unable to send read_info cmd\n"); + exit(EXIT_FAILURE); + } + } + + if (monitor && count > 0) + printf("\n"); +} + +static void cmd_info(int mgmt_sk, uint16_t index, int argc, char **argv) +{ + if (index == MGMT_INDEX_NONE) { + if (mgmt_send_cmd(mgmt_sk, MGMT_OP_READ_INDEX_LIST, + MGMT_INDEX_NONE, NULL, 0, + index_rsp, NULL) < 0) { + fprintf(stderr, "Unable to send index_list cmd\n"); + exit(EXIT_FAILURE); + } + + return; + } + + if (mgmt_send_cmd(mgmt_sk, MGMT_OP_READ_INFO, index, NULL, + 0, info_rsp, NULL) < 0) { + fprintf(stderr, "Unable to send read_info cmd\n"); + exit(EXIT_FAILURE); + } +} + +static void setting_rsp(int mgmt_sk, uint16_t op, uint16_t id, uint8_t status, + void *rsp, uint16_t len, void *user_data) +{ + uint32_t *rp = rsp; + + if (status != 0) { + fprintf(stderr, + "%s for hci%u failed with status 0x%02x (%s)\n", + mgmt_opstr(op), id, status, mgmt_errstr(status)); + exit(EXIT_FAILURE); + } + + if (len < sizeof(*rp)) { + fprintf(stderr, "Too small %s response (%u bytes)\n", + mgmt_opstr(op), len); + exit(EXIT_FAILURE); + } + + printf("hci%u %s complete, settings: ", id, mgmt_opstr(op)); + print_settings(bt_get_le32(rp)); + printf("\n"); + + exit(EXIT_SUCCESS); +} + +static void cmd_setting(int mgmt_sk, uint16_t index, uint16_t op, + int argc, char **argv) +{ + uint8_t val; + + if (argc < 2) { + printf("Specify \"on\" or \"off\"\n"); + exit(EXIT_FAILURE); + } + + if (strcasecmp(argv[1], "on") == 0 || strcasecmp(argv[1], "yes") == 0) + val = 1; + else if (strcasecmp(argv[1], "off") == 0) + val = 0; + else + val = atoi(argv[1]); + + if (index == MGMT_INDEX_NONE) + index = 0; + + if (mgmt_send_cmd(mgmt_sk, op, index, &val, sizeof(val), + setting_rsp, NULL) < 0) { + fprintf(stderr, "Unable to send %s cmd\n", mgmt_opstr(op)); + exit(EXIT_FAILURE); + } +} + +static void cmd_power(int mgmt_sk, uint16_t index, int argc, char **argv) +{ + cmd_setting(mgmt_sk, index, MGMT_OP_SET_POWERED, argc, argv); +} + +static void cmd_discov(int mgmt_sk, uint16_t index, int argc, char **argv) +{ + struct mgmt_cp_set_discoverable cp; + + if (argc < 2) { + printf("Usage: btmgmt %s <yes/no> [timeout]\n", argv[0]); + exit(EXIT_FAILURE); + } + + memset(&cp, 0, sizeof(cp)); + + if (strcasecmp(argv[1], "on") == 0 || strcasecmp(argv[1], "yes") == 0) + cp.val = 1; + else if (strcasecmp(argv[1], "off") == 0) + cp.val = 0; + else + cp.val = atoi(argv[1]); + + if (argc > 2) + cp.timeout = htobs(atoi(argv[2])); + + if (index == MGMT_INDEX_NONE) + index = 0; + + if (mgmt_send_cmd(mgmt_sk, MGMT_OP_SET_DISCOVERABLE, index, + &cp, sizeof(cp), setting_rsp, NULL) < 0) { + fprintf(stderr, "Unable to send set_discoverable cmd\n"); + exit(EXIT_FAILURE); + } +} + +static void cmd_connectable(int mgmt_sk, uint16_t index, int argc, char **argv) +{ + cmd_setting(mgmt_sk, index, MGMT_OP_SET_CONNECTABLE, argc, argv); +} + +static void cmd_pairable(int mgmt_sk, uint16_t index, int argc, char **argv) +{ + cmd_setting(mgmt_sk, index, MGMT_OP_SET_PAIRABLE, argc, argv); +} + +static void cmd_linksec(int mgmt_sk, uint16_t index, int argc, char **argv) +{ + cmd_setting(mgmt_sk, index, MGMT_OP_SET_LINK_SECURITY, argc, argv); +} + +static void cmd_ssp(int mgmt_sk, uint16_t index, int argc, char **argv) +{ + cmd_setting(mgmt_sk, index, MGMT_OP_SET_SSP, argc, argv); +} + +static void cmd_hs(int mgmt_sk, uint16_t index, int argc, char **argv) +{ + cmd_setting(mgmt_sk, index, MGMT_OP_SET_HS, argc, argv); +} + +static void cmd_le(int mgmt_sk, uint16_t index, int argc, char **argv) +{ + cmd_setting(mgmt_sk, index, MGMT_OP_SET_LE, argc, argv); +} + +static void class_rsp(int mgmt_sk, uint16_t op, uint16_t id, uint8_t status, + void *rsp, uint16_t len, void *user_data) +{ + struct mgmt_ev_class_of_dev_changed *rp = rsp; + + if (len == 0 && status != 0) { + fprintf(stderr, "%s failed, status 0x%02x (%s)\n", + mgmt_opstr(op), status, mgmt_errstr(status)); + exit(EXIT_FAILURE); + } + + if (len != sizeof(*rp)) { + fprintf(stderr, "Unexpected %s len %u\n", mgmt_opstr(op), len); + exit(EXIT_FAILURE); + } + + printf("%s succeeded. Class 0x%02x%02x%02x\n", mgmt_opstr(op), + rp->class_of_dev[2], rp->class_of_dev[1], rp->class_of_dev[0]); + + exit(EXIT_SUCCESS); +} + +static void cmd_class(int mgmt_sk, uint16_t index, int argc, char **argv) +{ + uint8_t class[2]; + + if (argc < 3) { + printf("Usage: btmgmt %s <major> <minor>\n", argv[0]); + exit(EXIT_FAILURE); + } + + class[0] = atoi(argv[1]); + class[1] = atoi(argv[2]); + + if (index == MGMT_INDEX_NONE) + index = 0; + + if (mgmt_send_cmd(mgmt_sk, MGMT_OP_SET_DEV_CLASS, index, + class, sizeof(class), class_rsp, NULL) < 0) { + fprintf(stderr, "Unable to send set_dev_class cmd\n"); + exit(EXIT_FAILURE); + } +} + +static void disconnect_rsp(int mgmt_sk, uint16_t op, uint16_t id, + uint8_t status, void *rsp, uint16_t len, + void *user_data) +{ + struct mgmt_rp_disconnect *rp = rsp; + char addr[18]; + + if (len == 0 && status != 0) { + fprintf(stderr, "Disconnect failed with status 0x%02x (%s)\n", + status, mgmt_errstr(status)); + exit(EXIT_FAILURE); + } + + if (len != sizeof(*rp)) { + fprintf(stderr, "Invalid disconnect response length (%u)\n", + len); + exit(EXIT_FAILURE); + } + + ba2str(&rp->addr.bdaddr, addr); + + if (status == 0) { + printf("%s disconnected\n", addr); + exit(EXIT_SUCCESS); + } else { + fprintf(stderr, + "Disconnecting %s failed with status 0x%02x (%s)\n", + addr, status, mgmt_errstr(status)); + exit(EXIT_FAILURE); + } +} + +static void cmd_disconnect(int mgmt_sk, uint16_t index, int argc, char **argv) +{ + struct mgmt_cp_disconnect cp; + + if (argc < 2) { + printf("Usage: btmgmt %s <address>\n", argv[0]); + exit(EXIT_FAILURE); + } + + str2ba(argv[1], &cp.addr.bdaddr); + + if (index == MGMT_INDEX_NONE) + index = 0; + + if (mgmt_send_cmd(mgmt_sk, MGMT_OP_DISCONNECT, index, + &cp, sizeof(cp), disconnect_rsp, NULL) < 0) { + fprintf(stderr, "Unable to send disconnect cmd\n"); + exit(EXIT_FAILURE); + } +} + +static void con_rsp(int mgmt_sk, uint16_t op, uint16_t id, uint8_t status, + void *rsp, uint16_t len, void *user_data) +{ + struct mgmt_rp_get_connections *rp = rsp; + uint16_t count, i; + + if (len < sizeof(*rp)) { + fprintf(stderr, "Too small (%u bytes) get_connections rsp\n", + len); + exit(EXIT_FAILURE); + } + + count = bt_get_le16(&rp->conn_count); + if (len != sizeof(*rp) + count * sizeof(struct mgmt_addr_info)) { + fprintf(stderr, "Invalid get_connections length " + " (count=%u, len=%u)\n", count, len); + exit(EXIT_FAILURE); + } + + for (i = 0; i < count; i++) { + char addr[18]; + + ba2str(&rp->addr[i].bdaddr, addr); + + printf("%s type %s\n", addr, typestr(rp->addr[i].type)); + } + + exit(EXIT_SUCCESS); +} + +static void cmd_con(int mgmt_sk, uint16_t index, int argc, char **argv) +{ + if (index == MGMT_INDEX_NONE) + index = 0; + + if (mgmt_send_cmd(mgmt_sk, MGMT_OP_GET_CONNECTIONS, index, NULL, 0, + con_rsp, NULL) < 0) { + fprintf(stderr, "Unable to send get_connections cmd\n"); + exit(EXIT_FAILURE); + } +} + +static void find_rsp(int mgmt_sk, uint16_t op, uint16_t id, uint8_t status, + void *rsp, uint16_t len, void *user_data) +{ + if (status != 0) { + fprintf(stderr, + "Unable to start discovery. status 0x%02x (%s)\n", + status, mgmt_errstr(status)); + exit(EXIT_FAILURE); + } + + printf("Discovery started\n"); + discovery = true; +} + +static void find_usage(void) +{ + printf("Usage: btmgmt find [-l|-b]>\n"); +} + +static struct option find_options[] = { + { "help", 0, 0, 'h' }, + { "le-only", 1, 0, 'l' }, + { "bredr-only", 1, 0, 'b' }, + { 0, 0, 0, 0 } +}; + +static void cmd_find(int mgmt_sk, uint16_t index, int argc, char **argv) +{ + struct mgmt_cp_start_discovery cp; + uint8_t type; + int opt; + + if (index == MGMT_INDEX_NONE) + index = 0; + + type = 0; + hci_set_bit(BDADDR_BREDR, &type); + hci_set_bit(BDADDR_LE_PUBLIC, &type); + hci_set_bit(BDADDR_LE_RANDOM, &type); + + while ((opt = getopt_long(argc, argv, "+lbh", find_options, + NULL)) != -1) { + switch (opt) { + case 'l': + hci_clear_bit(BDADDR_BREDR, &type); + hci_set_bit(BDADDR_LE_PUBLIC, &type); + hci_set_bit(BDADDR_LE_RANDOM, &type); + break; + case 'b': + hci_set_bit(BDADDR_BREDR, &type); + hci_clear_bit(BDADDR_LE_PUBLIC, &type); + hci_clear_bit(BDADDR_LE_RANDOM, &type); + break; + case 'h': + default: + find_usage(); + exit(EXIT_SUCCESS); + } + } + + argc -= optind; + argv += optind; + optind = 0; + + memset(&cp, 0, sizeof(cp)); + cp.type = type; + + if (mgmt_send_cmd(mgmt_sk, MGMT_OP_START_DISCOVERY, index, + &cp, sizeof(cp), find_rsp, NULL) < 0) { + fprintf(stderr, "Unable to send start_discovery cmd\n"); + exit(EXIT_FAILURE); + } +} + +static void name_rsp(int mgmt_sk, uint16_t op, uint16_t id, uint8_t status, + void *rsp, uint16_t len, void *user_data) +{ + if (status != 0) { + fprintf(stderr, "Unable to set local name. status 0x%02x (%s)", + status, mgmt_errstr(status)); + exit(EXIT_FAILURE); + } + + exit(EXIT_SUCCESS); +} + +static void cmd_name(int mgmt_sk, uint16_t index, int argc, char **argv) +{ + struct mgmt_cp_set_local_name cp; + + if (argc < 2) { + printf("Usage: btmgmt %s <name> [shortname]\n", argv[0]); + exit(EXIT_FAILURE); + } + + if (index == MGMT_INDEX_NONE) + index = 0; + + memset(&cp, 0, sizeof(cp)); + strncpy((char *) cp.name, argv[1], HCI_MAX_NAME_LENGTH); + if (argc > 2) + strncpy((char *) cp.short_name, argv[2], + MGMT_MAX_SHORT_NAME_LENGTH); + + if (mgmt_send_cmd(mgmt_sk, MGMT_OP_SET_LOCAL_NAME, index, + &cp, sizeof(cp), name_rsp, NULL) < 0) { + fprintf(stderr, "Unable to send set_name cmd\n"); + exit(EXIT_FAILURE); + } +} + +static void pair_rsp(int mgmt_sk, uint16_t op, uint16_t id, uint8_t status, + void *rsp, uint16_t len, void *user_data) +{ + struct mgmt_rp_pair_device *rp = rsp; + char addr[18]; + + if (len == 0 && status != 0) { + fprintf(stderr, "Pairing failed with status 0x%02x (%s)\n", + status, mgmt_errstr(status)); + exit(EXIT_FAILURE); + } + + if (len != sizeof(*rp)) { + fprintf(stderr, "Unexpected pair_rsp len %u\n", len); + exit(EXIT_FAILURE); + } + + ba2str(&rp->addr.bdaddr, addr); + + if (status != 0) { + fprintf(stderr, + "Pairing with %s (%s) failed. status 0x%02x (%s)\n", + addr, typestr(rp->addr.type), status, + mgmt_errstr(status)); + exit(EXIT_FAILURE); + } + + printf("Paired with %s\n", addr); + + exit(EXIT_SUCCESS); +} + +static void pair_usage(void) +{ + printf("Usage: btmgmt pair [-c cap] [-t type] <remote address>\n"); +} + +static struct option pair_options[] = { + { "help", 0, 0, 'h' }, + { "capability", 1, 0, 'c' }, + { "type", 1, 0, 't' }, + { 0, 0, 0, 0 } +}; + +static void cmd_pair(int mgmt_sk, uint16_t index, int argc, char **argv) +{ + struct mgmt_cp_pair_device cp; + uint8_t cap = 0x01; + uint8_t type = BDADDR_BREDR; + int opt; + + while ((opt = getopt_long(argc, argv, "+c:t:h", pair_options, + NULL)) != -1) { + switch (opt) { + case 'c': + cap = strtol(optarg, NULL, 0); + break; + case 't': + type = strtol(optarg, NULL, 0); + break; + case 'h': + default: + pair_usage(); + exit(EXIT_SUCCESS); + } + } + + argc -= optind; + argv += optind; + optind = 0; + + if (argc < 1) { + pair_usage(); + exit(EXIT_FAILURE); + } + + if (index == MGMT_INDEX_NONE) + index = 0; + + memset(&cp, 0, sizeof(cp)); + str2ba(argv[0], &cp.addr.bdaddr); + cp.addr.type = type; + cp.io_cap = cap; + + if (mgmt_send_cmd(mgmt_sk, MGMT_OP_PAIR_DEVICE, index, &cp, sizeof(cp), + pair_rsp, NULL) < 0) { + fprintf(stderr, "Unable to send pair_device cmd\n"); + exit(EXIT_FAILURE); + } +} + +static void unpair_rsp(int mgmt_sk, uint16_t op, uint16_t id, uint8_t status, + void *rsp, uint16_t len, void *user_data) +{ + struct mgmt_rp_unpair_device *rp = rsp; + char addr[18]; + + if (len == 0 && status != 0) { + fprintf(stderr, "Unpair device failed. status 0x%02x (%s)\n", + status, mgmt_errstr(status)); + exit(EXIT_FAILURE); + } + + if (len != sizeof(*rp)) { + fprintf(stderr, "Unexpected unpair_device_rsp len %u\n", len); + exit(EXIT_FAILURE); + } + + ba2str(&rp->addr.bdaddr, addr); + + if (status != 0) { + fprintf(stderr, + "Unpairing %s failed. status 0x%02x (%s)\n", + addr, status, mgmt_errstr(status)); + exit(EXIT_FAILURE); + } + + printf("%s unpaired\n", addr); + + exit(EXIT_SUCCESS); +} + +static void cmd_unpair(int mgmt_sk, uint16_t index, int argc, char **argv) +{ + struct mgmt_cp_unpair_device cp; + + if (argc < 2) { + printf("Usage: btmgmt %s <remote address>\n", argv[0]); + exit(EXIT_FAILURE); + } + + if (index == MGMT_INDEX_NONE) + index = 0; + + memset(&cp, 0, sizeof(cp)); + str2ba(argv[1], &cp.addr.bdaddr); + cp.disconnect = 1; + + if (mgmt_send_cmd(mgmt_sk, MGMT_OP_UNPAIR_DEVICE, index, &cp, + sizeof(cp), unpair_rsp, NULL) < 0) { + fprintf(stderr, "Unable to send unpair_device cmd\n"); + exit(EXIT_FAILURE); + } +} + +static void keys_rsp(int mgmt_sk, uint16_t op, uint16_t id, uint8_t status, + void *rsp, uint16_t len, void *user_data) +{ + if (status != 0) { + fprintf(stderr, "Load keys failed with status 0x%02x (%s)\n", + status, mgmt_errstr(status)); + exit(EXIT_FAILURE); + } + + printf("Keys successfully loaded\n"); + + exit(EXIT_SUCCESS); +} + +static void cmd_keys(int mgmt_sk, uint16_t index, int argc, char **argv) +{ + struct mgmt_cp_load_link_keys cp; + + if (index == MGMT_INDEX_NONE) + index = 0; + + memset(&cp, 0, sizeof(cp)); + + if (mgmt_send_cmd(mgmt_sk, MGMT_OP_LOAD_LINK_KEYS, index, + &cp, sizeof(cp), keys_rsp, NULL) < 0) { + fprintf(stderr, "Unable to send load_keys cmd\n"); + exit(EXIT_FAILURE); + } +} + +static void block_rsp(int mgmt_sk, uint16_t op, uint16_t id, uint8_t status, + void *rsp, uint16_t len, void *user_data) +{ + struct mgmt_addr_info *rp = rsp; + char addr[18]; + + if (len == 0 && status != 0) { + fprintf(stderr, "%s failed, status 0x%02x (%s)\n", + mgmt_opstr(op), status, mgmt_errstr(status)); + exit(EXIT_FAILURE); + } + + if (len != sizeof(*rp)) { + fprintf(stderr, "Unexpected %s len %u\n", mgmt_opstr(op), len); + exit(EXIT_FAILURE); + } + + ba2str(&rp->bdaddr, addr); + + if (status != 0) { + fprintf(stderr, "%s %s (%s) failed. status 0x%02x (%s)\n", + mgmt_opstr(op), addr, typestr(rp->type), + status, mgmt_errstr(status)); + exit(EXIT_FAILURE); + } + + printf("%s %s succeeded\n", mgmt_opstr(op), addr); + + exit(EXIT_SUCCESS); +} + +static void block_usage(void) +{ + printf("Usage: btmgmt block [-t type] <remote address>\n"); +} + +static struct option block_options[] = { + { "help", 0, 0, 'h' }, + { "type", 1, 0, 't' }, + { 0, 0, 0, 0 } +}; + +static void cmd_block(int mgmt_sk, uint16_t index, int argc, char **argv) +{ + struct mgmt_cp_block_device cp; + uint8_t type = BDADDR_BREDR; + int opt; + + while ((opt = getopt_long(argc, argv, "+t:h", block_options, + NULL)) != -1) { + switch (opt) { + case 't': + type = strtol(optarg, NULL, 0); + break; + case 'h': + default: + block_usage(); + exit(EXIT_SUCCESS); + } + } + + argc -= optind; + argv += optind; + optind = 0; + + if (argc < 1) { + block_usage(); + exit(EXIT_FAILURE); + } + + if (index == MGMT_INDEX_NONE) + index = 0; + + memset(&cp, 0, sizeof(cp)); + str2ba(argv[0], &cp.addr.bdaddr); + cp.addr.type = type; + + if (mgmt_send_cmd(mgmt_sk, MGMT_OP_BLOCK_DEVICE, index, + &cp, sizeof(cp), block_rsp, NULL) < 0) { + fprintf(stderr, "Unable to send block_device cmd\n"); + exit(EXIT_FAILURE); + } +} + +static void unblock_usage(void) +{ + printf("Usage: btmgmt unblock [-t type] <remote address>\n"); +} + +static void cmd_unblock(int mgmt_sk, uint16_t index, int argc, char **argv) +{ + struct mgmt_cp_unblock_device cp; + uint8_t type = BDADDR_BREDR; + int opt; + + while ((opt = getopt_long(argc, argv, "+t:h", block_options, + NULL)) != -1) { + switch (opt) { + case 't': + type = strtol(optarg, NULL, 0); + break; + case 'h': + default: + unblock_usage(); + exit(EXIT_SUCCESS); + } + } + + argc -= optind; + argv += optind; + optind = 0; + + if (argc < 1) { + unblock_usage(); + exit(EXIT_FAILURE); + } + + if (index == MGMT_INDEX_NONE) + index = 0; + + memset(&cp, 0, sizeof(cp)); + str2ba(argv[0], &cp.addr.bdaddr); + cp.addr.type = type; + + if (mgmt_send_cmd(mgmt_sk, MGMT_OP_UNBLOCK_DEVICE, index, + &cp, sizeof(cp), block_rsp, NULL) < 0) { + fprintf(stderr, "Unable to send unblock_device cmd\n"); + exit(EXIT_FAILURE); + } +} + +static void uuid_to_uuid128(uuid_t *uuid128, const uuid_t *uuid) +{ + if (uuid->type == SDP_UUID16) + sdp_uuid16_to_uuid128(uuid128, uuid); + else if (uuid->type == SDP_UUID32) + sdp_uuid32_to_uuid128(uuid128, uuid); + else + memcpy(uuid128, uuid, sizeof(*uuid)); +} + +static void cmd_add_uuid(int mgmt_sk, uint16_t index, int argc, char **argv) +{ + struct mgmt_cp_add_uuid cp; + uint128_t uint128; + uuid_t uuid, uuid128; + + if (argc < 3) { + printf("UUID and service hint needed\n"); + exit(EXIT_FAILURE); + } + + if (index == MGMT_INDEX_NONE) + index = 0; + + if (bt_string2uuid(&uuid, argv[1]) < 0) { + printf("Invalid UUID: %s\n", argv[1]); + exit(EXIT_FAILURE); + } + + memset(&cp, 0, sizeof(cp)); + + uuid_to_uuid128(&uuid128, &uuid); + ntoh128((uint128_t *) uuid128.value.uuid128.data, &uint128); + htob128(&uint128, (uint128_t *) cp.uuid); + + cp.svc_hint = atoi(argv[2]); + + if (mgmt_send_cmd(mgmt_sk, MGMT_OP_ADD_UUID, index, + &cp, sizeof(cp), class_rsp, NULL) < 0) { + fprintf(stderr, "Unable to send add_uuid cmd\n"); + exit(EXIT_FAILURE); + } +} + +static void cmd_remove_uuid(int mgmt_sk, uint16_t index, int argc, char **argv) +{ + struct mgmt_cp_remove_uuid cp; + uint128_t uint128; + uuid_t uuid, uuid128; + + if (argc < 2) { + printf("UUID needed\n"); + exit(EXIT_FAILURE); + } + + if (index == MGMT_INDEX_NONE) + index = 0; + + if (bt_string2uuid(&uuid, argv[1]) < 0) { + printf("Invalid UUID: %s\n", argv[1]); + exit(EXIT_FAILURE); + } + + memset(&cp, 0, sizeof(cp)); + + uuid_to_uuid128(&uuid128, &uuid); + ntoh128((uint128_t *) uuid128.value.uuid128.data, &uint128); + htob128(&uint128, (uint128_t *) cp.uuid); + + if (mgmt_send_cmd(mgmt_sk, MGMT_OP_REMOVE_UUID, index, + &cp, sizeof(cp), class_rsp, NULL) < 0) { + fprintf(stderr, "Unable to send remove_uuid cmd\n"); + exit(EXIT_FAILURE); + } +} + +static void cmd_clr_uuids(int mgmt_sk, uint16_t index, int argc, char **argv) +{ + char *uuid_any = "00000000-0000-0000-0000-000000000000"; + char *rm_argv[] = { "rm-uuid", uuid_any, NULL }; + + cmd_remove_uuid(mgmt_sk, index, 2, rm_argv); +} + +static void did_rsp(int mgmt_sk, uint16_t op, uint16_t id, uint8_t status, + void *rsp, uint16_t len, void *user_data) +{ + if (status != 0) { + fprintf(stderr, "Set Device ID failed with status 0x%02x (%s)\n", + status, mgmt_errstr(status)); + exit(EXIT_FAILURE); + } + + printf("Device ID successfully set\n"); + + exit(EXIT_SUCCESS); +} + +static void did_usage(void) +{ + printf("Usage: btmgmt did <source>:<vendor>:<product>:<version>\n"); + printf(" possible source values: bluetooth, usb\n"); +} + +static void cmd_did(int mgmt_sk, uint16_t index, int argc, char **argv) +{ + struct mgmt_cp_set_device_id cp; + uint16_t vendor, product, version , source; + int result; + + if (argc < 2) { + did_usage(); + exit(EXIT_FAILURE); + } + + result = sscanf(argv[1], "bluetooth:%4hx:%4hx:%4hx", &vendor, &product, + &version); + if (result == 3) { + source = 0x0001; + goto done; + } + + result = sscanf(argv[1], "usb:%4hx:%4hx:%4hx", &vendor, &product, + &version); + if (result == 3) { + source = 0x0002; + goto done; + } + + did_usage(); + exit(EXIT_FAILURE); + +done: + if (index == MGMT_INDEX_NONE) + index = 0; + + cp.source = htobs(source); + cp.vendor = htobs(vendor); + cp.product = htobs(product); + cp.version = htobs(version); + + if (mgmt_send_cmd(mgmt_sk, MGMT_OP_SET_DEVICE_ID, index, + &cp, sizeof(cp), did_rsp, NULL) < 0) { + fprintf(stderr, "Unable to send set_dev_class cmd\n"); + exit(EXIT_FAILURE); + } +} + +static struct { + char *cmd; + void (*func)(int mgmt_sk, uint16_t index, int argc, char **argv); + char *doc; +} command[] = { + { "monitor", cmd_monitor, "Monitor events" }, + { "version", cmd_version, "Get the MGMT Version" }, + { "commands", cmd_commands, "List supported commands" }, + { "info", cmd_info, "Show controller info" }, + { "power", cmd_power, "Toggle powered state" }, + { "discov", cmd_discov, "Toggle discoverable state" }, + { "connectable",cmd_connectable,"Toggle connectable state" }, + { "pairable", cmd_pairable, "Toggle pairable state" }, + { "linksec", cmd_linksec, "Toggle link level security" }, + { "ssp", cmd_ssp, "Toggle SSP mode" }, + { "hs", cmd_hs, "Toggle HS Support" }, + { "le", cmd_le, "Toggle LE Support" }, + { "class", cmd_class, "Set device major/minor class" }, + { "disconnect", cmd_disconnect, "Disconnect device" }, + { "con", cmd_con, "List connections" }, + { "find", cmd_find, "Discover nearby devices" }, + { "name", cmd_name, "Set local name" }, + { "pair", cmd_pair, "Pair with a remote device" }, + { "unpair", cmd_unpair, "Unpair device" }, + { "keys", cmd_keys, "Load Keys" }, + { "block", cmd_block, "Block Device" }, + { "unblock", cmd_unblock, "Unblock Device" }, + { "add-uuid", cmd_add_uuid, "Add UUID" }, + { "rm-uuid", cmd_add_uuid, "Remove UUID" }, + { "clr-uuids", cmd_clr_uuids, "Clear UUIDs", }, + { "did", cmd_did, "Set Device ID", }, + { NULL, NULL, 0 } +}; + +static void usage(void) +{ + int i; + + printf("btmgmt ver %s\n", VERSION); + printf("Usage:\n" + "\tbtmgmt [options] <command> [command parameters]\n"); + + printf("Options:\n" + "\t--index <id>\tSpecify adapter index\n" + "\t--verbose\tEnable extra logging\n" + "\t--help\tDisplay help\n"); + + printf("Commands:\n"); + for (i = 0; command[i].cmd; i++) + printf("\t%-15s\t%s\n", command[i].cmd, command[i].doc); + + printf("\n" + "For more information on the usage of each command use:\n" + "\tbtmgmt <command> --help\n" ); +} + +static struct option main_options[] = { + { "index", 1, 0, 'i' }, + { "verbose", 0, 0, 'v' }, + { "help", 0, 0, 'h' }, + { 0, 0, 0, 0 } +}; + +int main(int argc, char *argv[]) +{ + int opt, i, mgmt_sk; + uint16_t index = MGMT_INDEX_NONE; + struct pollfd pollfd; + + while ((opt = getopt_long(argc, argv, "+hvi:", + main_options, NULL)) != -1) { + switch (opt) { + case 'i': + if (strlen(optarg) > 3 && + strncasecmp(optarg, "hci", 3) == 0) + index = atoi(&optarg[4]); + else + index = atoi(optarg); + break; + case 'v': + monitor = true; + break; + case 'h': + default: + usage(); + return 0; + } + } + + argc -= optind; + argv += optind; + optind = 0; + + if (argc < 1) { + usage(); + return 0; + } + + mgmt_sk = mgmt_open(); + if (mgmt_sk < 0) { + fprintf(stderr, "Unable to open mgmt socket\n"); + return -1; + } + + for (i = 0; command[i].cmd; i++) { + if (strcmp(command[i].cmd, argv[0]) != 0) + continue; + + command[i].func(mgmt_sk, index, argc, argv); + break; + } + + if (command[i].cmd == NULL) { + fprintf(stderr, "Unknown command: %s\n", argv[0]); + close(mgmt_sk); + return -1; + } + + pollfd.fd = mgmt_sk; + pollfd.events = POLLIN; + pollfd.revents = 0; + + while (poll(&pollfd, 1, -1) >= 0) { + if (pollfd.revents & (POLLHUP | POLLERR | POLLNVAL)) + break; + + if (pollfd.revents & POLLIN) + mgmt_process_data(mgmt_sk); + + pollfd.revents = 0; + } + + close(mgmt_sk); + + return 0; +} -- 1.7.10.2 -- To unsubscribe from this list: send the line "unsubscribe linux-bluetooth" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html