| /* |
| * Licensed to the Apache Software Foundation (ASF) under one |
| * or more contributor license agreements. See the NOTICE file |
| * distributed with this work for additional information |
| * regarding copyright ownership. The ASF licenses this file |
| * to you under the Apache License, Version 2.0 (the |
| * "License"); you may not use this file except in compliance |
| * with the License. You may obtain a copy of the License at |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, |
| * software distributed under the License is distributed on an |
| * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY |
| * KIND, either express or implied. See the License for the |
| * specific language governing permissions and limitations |
| * under the License. |
| */ |
| |
| #include <assert.h> |
| #include <inttypes.h> |
| #include <errno.h> |
| #include <string.h> |
| #include "os/mynewt.h" |
| #include "bsp/bsp.h" |
| |
| #include "nimble/ble.h" |
| #include "nimble/nimble_opt.h" |
| #include "nimble/hci_common.h" |
| #include "host/ble_gap.h" |
| #include "host/ble_hs_adv.h" |
| #include "host/ble_sm.h" |
| #include "host/ble_eddystone.h" |
| #include "host/ble_hs_id.h" |
| #include "services/gatt/ble_svc_gatt.h" |
| #include "../src/ble_hs_priv.h" |
| |
| #include "console/console.h" |
| #include "shell/shell.h" |
| |
| #include "cmd.h" |
| #include "btshell.h" |
| #include "cmd_gatt.h" |
| #include "cmd_iso.h" |
| #include "cmd_l2cap.h" |
| #include "cmd_leaudio.h" |
| |
| #define BTSHELL_MODULE "btshell" |
| |
| int |
| cmd_parse_conn_start_end(uint16_t *out_conn, uint16_t *out_start, |
| uint16_t *out_end) |
| { |
| int rc; |
| |
| *out_conn = parse_arg_uint16("conn", &rc); |
| if (rc != 0) { |
| return rc; |
| } |
| |
| *out_start = parse_arg_uint16("start", &rc); |
| if (rc != 0) { |
| return rc; |
| } |
| |
| *out_end = parse_arg_uint16("end", &rc); |
| if (rc != 0) { |
| return rc; |
| } |
| |
| return 0; |
| } |
| |
| static const struct parse_arg_kv_pair cmd_own_addr_types[] = { |
| { "public", BLE_OWN_ADDR_PUBLIC }, |
| { "random", BLE_OWN_ADDR_RANDOM }, |
| { "rpa_pub", BLE_OWN_ADDR_RPA_PUBLIC_DEFAULT }, |
| { "rpa_rnd", BLE_OWN_ADDR_RPA_RANDOM_DEFAULT }, |
| { NULL } |
| }; |
| |
| static const struct parse_arg_kv_pair cmd_peer_addr_types[] = { |
| { "public", BLE_ADDR_PUBLIC }, |
| { "random", BLE_ADDR_RANDOM }, |
| { "public_id", BLE_ADDR_PUBLIC_ID }, |
| { "random_id", BLE_ADDR_RANDOM_ID }, |
| { NULL } |
| }; |
| |
| static const struct parse_arg_kv_pair cmd_addr_type[] = { |
| { "public", BLE_ADDR_PUBLIC }, |
| { "random", BLE_ADDR_RANDOM }, |
| { NULL } |
| }; |
| |
| |
| static int |
| parse_dev_addr(const char *prefix, const struct parse_arg_kv_pair *addr_types, |
| ble_addr_t *addr) |
| { |
| char name[32]; |
| int rc; |
| int written = 0; |
| |
| if (!prefix) { |
| name[0] = '\0'; |
| } else { |
| written = snprintf(name, sizeof(name), "%s", prefix); |
| if (written >= sizeof(name) || written < 0) { |
| return EINVAL; |
| } |
| } |
| |
| rc = snprintf(name + written, sizeof(name) - written, "%s", "addr"); |
| if (rc >= sizeof(name) - written || rc < 0) { |
| return EINVAL; |
| } |
| written += rc; |
| |
| rc = parse_arg_ble_addr(name, addr); |
| if (rc == ENOENT) { |
| /* not found */ |
| return rc; |
| } else if (rc == EAGAIN) { |
| /* address found, but no type provided */ |
| rc = written; |
| written = snprintf(name + written, sizeof(name) - written, "%s", "_type"); |
| if (written >= sizeof(name) - rc || written < 0) { |
| return EINVAL; |
| } |
| addr->type = parse_arg_kv(name, addr_types, &rc); |
| if (rc == ENOENT) { |
| addr->type = BLE_ADDR_PUBLIC; |
| } else if (rc != 0) { |
| return rc; |
| } |
| } else if (rc != 0) { |
| /* error parsing address */ |
| return rc; |
| } else { |
| /* full address found, but let's just make sure there is no type arg */ |
| rc = written; |
| written = snprintf(name + written, sizeof(name) - written, "%s", "_type"); |
| if (written >= sizeof(name) - rc || written < 0) { |
| return EINVAL; |
| } |
| if (parse_arg_extract(name)) { |
| return E2BIG; |
| } |
| } |
| |
| return 0; |
| } |
| |
| /***************************************************************************** |
| * $advertise * |
| *****************************************************************************/ |
| static const struct parse_arg_kv_pair cmd_adv_filt_types[] = { |
| { "none", BLE_HCI_ADV_FILT_NONE }, |
| { "scan", BLE_HCI_ADV_FILT_SCAN }, |
| { "conn", BLE_HCI_ADV_FILT_CONN }, |
| { "both", BLE_HCI_ADV_FILT_BOTH }, |
| { NULL } |
| }; |
| |
| #if MYNEWT_VAL(BLE_EXT_ADV) |
| static struct parse_arg_kv_pair cmd_ext_adv_phy_opts[] = { |
| { "1M", 0x01 }, |
| { "2M", 0x02 }, |
| { "coded", 0x03 }, |
| { NULL } |
| }; |
| |
| static int |
| cmd_advertise_configure(int argc, char **argv) |
| { |
| struct ble_gap_ext_adv_params params = {0}; |
| int8_t selected_tx_power; |
| uint8_t instance; |
| int rc; |
| |
| rc = parse_arg_init(argc - 1, argv + 1); |
| if (rc != 0) { |
| return rc; |
| } |
| |
| instance = parse_arg_uint8_dflt("instance", 0, &rc); |
| if (rc != 0 || instance >= BLE_ADV_INSTANCES) { |
| console_printf("invalid instance\n"); |
| return rc; |
| } |
| |
| memset(¶ms, 0, sizeof(params)); |
| |
| params.legacy_pdu = parse_arg_bool_dflt("legacy", 0, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'legacy' parameter\n"); |
| return rc; |
| } |
| |
| if (params.legacy_pdu) { |
| params.connectable = 1; |
| params.scannable = 1; |
| } |
| |
| params.connectable = parse_arg_bool_dflt("connectable", params.connectable, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'connectable' parameter\n"); |
| return rc; |
| } |
| |
| params.scannable = parse_arg_bool_dflt("scannable", params.scannable, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'scannable' parameter\n"); |
| return rc; |
| } |
| |
| params.high_duty_directed = parse_arg_bool_dflt("high_duty", 0, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'high_duty' parameter\n"); |
| return rc; |
| } |
| |
| params.anonymous = parse_arg_bool_dflt("anonymous", 0, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'anonymous' parameter\n"); |
| return rc; |
| } |
| |
| params.include_tx_power = parse_arg_bool_dflt("include_tx_power", 0, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'include_tx_power' parameter\n"); |
| return rc; |
| } |
| |
| params.scan_req_notif = parse_arg_bool_dflt("scan_req_notif", 0, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'scan_req_notif' parameter\n"); |
| return rc; |
| } |
| |
| rc = parse_dev_addr("peer_", cmd_peer_addr_types, ¶ms.peer); |
| if (rc == 0) { |
| params.directed = 1; |
| } else if (rc == ENOENT) { |
| /* skip, no peer address provided */ |
| } else { |
| console_printf("invalid 'peer_addr' parameter\n"); |
| return rc; |
| } |
| |
| |
| params.directed = parse_arg_bool_dflt("directed", params.directed, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'directed' parameter\n"); |
| return rc; |
| } |
| |
| if (params.directed && params.legacy_pdu) { |
| params.scannable = 0; |
| } |
| |
| params.own_addr_type = parse_arg_kv_dflt("own_addr_type", |
| cmd_own_addr_types, |
| btshell_get_default_own_addr_type(), &rc); |
| if (rc != 0) { |
| console_printf("invalid 'own_addr_type' parameter\n"); |
| return rc; |
| } |
| |
| params.channel_map = parse_arg_uint8_dflt("channel_map", 0, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'channel_map' parameter\n"); |
| return rc; |
| } |
| |
| params.filter_policy = parse_arg_kv_dflt("filter", cmd_adv_filt_types, |
| BLE_HCI_ADV_FILT_NONE, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'filter' parameter\n"); |
| return rc; |
| } |
| |
| params.itvl_min = parse_arg_time_dflt("interval_min", 625, 0, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'interval_min' parameter\n"); |
| return rc; |
| } |
| |
| params.itvl_max = parse_arg_time_dflt("interval_max", 625, 0, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'interval_max' parameter\n"); |
| return rc; |
| } |
| |
| params.tx_power = parse_arg_long_bounds_dflt("tx_power", |
| -127, 127, 127, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'tx_power' parameter\n"); |
| return rc; |
| } |
| |
| params.primary_phy = parse_arg_kv_dflt("primary_phy", cmd_ext_adv_phy_opts, |
| 1, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'primary_phy' parameter\n"); |
| return rc; |
| } |
| |
| params.secondary_phy = parse_arg_kv_dflt("secondary_phy", |
| cmd_ext_adv_phy_opts, |
| params.primary_phy, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'secondary_phy' parameter\n"); |
| return rc; |
| } |
| |
| params.sid = parse_arg_uint8_dflt("sid", 0, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'sid' parameter\n"); |
| return rc; |
| } |
| |
| rc = btshell_ext_adv_configure(instance, ¶ms, &selected_tx_power); |
| if (rc) { |
| console_printf("failed to configure advertising instance\n"); |
| return rc; |
| } |
| |
| console_printf("Instance %u configured (selected tx power: %d)\n", |
| instance, selected_tx_power); |
| |
| return 0; |
| } |
| |
| static int |
| cmd_advertise_set_addr(int argc, char **argv) |
| { |
| ble_addr_t addr; |
| uint8_t instance; |
| int rc; |
| |
| rc = parse_arg_init(argc - 1, argv + 1); |
| if (rc != 0) { |
| return rc; |
| } |
| |
| instance = parse_arg_uint8_dflt("instance", 0, &rc); |
| if (rc != 0 || instance >= BLE_ADV_INSTANCES) { |
| console_printf("invalid instance\n"); |
| return rc; |
| } |
| |
| rc = parse_arg_mac_addr("addr", addr.val); |
| if (rc != 0) { |
| console_printf("invalid 'addr' parameter\n"); |
| return rc; |
| } |
| |
| addr.type = BLE_ADDR_RANDOM; |
| |
| rc = ble_gap_ext_adv_set_addr(instance, &addr); |
| if (rc) { |
| console_printf("failed to start advertising instance\n"); |
| return rc; |
| } |
| |
| return 0; |
| } |
| |
| static int |
| cmd_advertise_start(int argc, char **argv) |
| { |
| int max_events; |
| uint8_t instance; |
| int duration; |
| bool restart; |
| int rc; |
| |
| rc = parse_arg_init(argc - 1, argv + 1); |
| if (rc != 0) { |
| return rc; |
| } |
| |
| instance = parse_arg_uint8_dflt("instance", 0, &rc); |
| if (rc != 0 || instance >= BLE_ADV_INSTANCES) { |
| console_printf("invalid instance\n"); |
| return rc; |
| } |
| |
| duration = parse_arg_uint16_dflt("duration", 0, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'duration' parameter\n"); |
| return rc; |
| } |
| |
| max_events = parse_arg_uint8_dflt("max_events", 0, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'max_events' parameter\n"); |
| return rc; |
| } |
| |
| restart = parse_arg_bool_dflt("restart", 0, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'restart' parameter\n"); |
| return rc; |
| } |
| |
| rc = btshell_ext_adv_start(instance, duration, max_events, restart); |
| if (rc) { |
| console_printf("failed to start advertising instance\n"); |
| return rc; |
| } |
| |
| return 0; |
| } |
| |
| static int |
| cmd_advertise_stop(int argc, char **argv) |
| { |
| uint8_t instance; |
| int rc; |
| |
| rc = parse_arg_init(argc - 1, argv + 1); |
| if (rc != 0) { |
| return rc; |
| } |
| |
| instance = parse_arg_uint8_dflt("instance", 0, &rc); |
| if (rc != 0 || instance >= BLE_ADV_INSTANCES) { |
| console_printf("invalid instance\n"); |
| return rc; |
| } |
| |
| rc = btshell_ext_adv_stop(instance); |
| if (rc) { |
| console_printf("failed to stop advertising instance\n"); |
| return rc; |
| } |
| |
| return 0; |
| } |
| |
| static int |
| cmd_advertise_remove(int argc, char **argv) |
| { |
| uint8_t instance; |
| int rc; |
| |
| rc = parse_arg_init(argc - 1, argv + 1); |
| if (rc != 0) { |
| return rc; |
| } |
| |
| instance = parse_arg_uint8_dflt("instance", 0, &rc); |
| if (rc != 0 || instance >= BLE_ADV_INSTANCES) { |
| console_printf("invalid instance\n"); |
| return rc; |
| } |
| |
| rc = ble_gap_ext_adv_remove(instance); |
| if (rc) { |
| console_printf("failed to remove advertising instance\n"); |
| return rc; |
| } |
| |
| return 0; |
| } |
| |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| static const struct shell_param advertise_configure_params[] = { |
| {"instance", "default: 0"}, |
| {"connectable", "connectable advertising, usage: =[0-1], default: 0"}, |
| {"scannable", "scannable advertising, usage: =[0-1], default: 0"}, |
| {"directed", "directed advertising, usage: =[0-1], default: 0"}, |
| {"peer_addr_type", "usage: =[public|random|public_id|random_id], default: public"}, |
| {"peer_addr", "usage: =[XX:XX:XX:XX:XX:XX]"}, |
| {"own_addr_type", "usage: =[public|random|rpa_pub|rpa_rnd], default: public if available, otherwise random"}, |
| {"channel_map", "usage: =[0x00-0xff], default: 0"}, |
| {"filter", "usage: =[none|scan|conn|both], default: none"}, |
| {"interval_min", "usage: =[0-UINT32_MAX], default: 0"}, |
| {"interval_max", "usage: =[0-UINT32_MAX], default: 0"}, |
| {"tx_power", "usage: =[-127-127], default: 127"}, |
| {"primary_phy", "usage: =[1M|coded], default: 1M"}, |
| {"secondary_phy", "usage: =[1M|2M|coded], default: primary_phy"}, |
| {"sid", "usage: =[0-UINT8_MAX], default: 0"}, |
| {"high_duty", "usage: =[0-1], default: 0"}, |
| {"anonymous", "enable anonymous advertising, usage: =[0-1], default: 0"}, |
| {"legacy", "use legacy PDUs, usage: =[0-1], default: 0"}, |
| {"include_tx_power", "include TX power in PDU, usage: =[0-1], default: 0"}, |
| {"scan_req_notif", "enable Scan Request notification usage: =[0-1], default: 0"}, |
| {NULL, NULL} |
| }; |
| |
| static const struct shell_cmd_help advertise_configure_help = { |
| .summary = "configure new advertising instance", |
| .usage = NULL, |
| .params = advertise_configure_params, |
| }; |
| |
| static const struct shell_param advertise_set_addr_params[] = { |
| {"instance", "default: 0"}, |
| {"addr", "usage: =[XX:XX:XX:XX:XX:XX]"}, |
| {NULL, NULL} |
| }; |
| |
| static const struct shell_cmd_help advertise_set_addr_help = { |
| .summary = "set advertising instance random address", |
| .usage = NULL, |
| .params = advertise_set_addr_params, |
| }; |
| |
| static const struct shell_param advertise_start_params[] = { |
| {"instance", "default: 0"}, |
| {"duration", "advertising duration in 10ms units, default: 0 (forever)"}, |
| {"max_events", "max number of advertising events, default: 0 (no limit)"}, |
| {"restart", "restart advertising after disconnect, usage: =[0-1], default: 0"}, |
| {NULL, NULL} |
| }; |
| |
| static const struct shell_cmd_help advertise_start_help = { |
| .summary = "start advertising instance", |
| .usage = NULL, |
| .params = advertise_start_params, |
| }; |
| |
| static const struct shell_param advertise_stop_params[] = { |
| {"instance", "default: 0"}, |
| {NULL, NULL} |
| }; |
| |
| static const struct shell_cmd_help advertise_stop_help = { |
| .summary = "stop advertising instance", |
| .usage = NULL, |
| .params = advertise_stop_params, |
| }; |
| |
| static const struct shell_param advertise_remove_params[] = { |
| {"instance", "default: 0"}, |
| {NULL, NULL} |
| }; |
| |
| static const struct shell_cmd_help advertise_remove_help = { |
| .summary = "remove advertising instance", |
| .usage = NULL, |
| .params = advertise_remove_params, |
| }; |
| #endif |
| |
| #else |
| static const struct parse_arg_kv_pair cmd_adv_conn_modes[] = { |
| { "non", BLE_GAP_CONN_MODE_NON }, |
| { "und", BLE_GAP_CONN_MODE_UND }, |
| { "dir", BLE_GAP_CONN_MODE_DIR }, |
| { NULL } |
| }; |
| |
| static const struct parse_arg_kv_pair cmd_adv_disc_modes[] = { |
| { "non", BLE_GAP_DISC_MODE_NON }, |
| { "ltd", BLE_GAP_DISC_MODE_LTD }, |
| { "gen", BLE_GAP_DISC_MODE_GEN }, |
| { NULL } |
| }; |
| |
| static int |
| cmd_advertise(int argc, char **argv) |
| { |
| struct ble_gap_adv_params params; |
| int32_t duration_ms; |
| ble_addr_t peer_addr; |
| ble_addr_t *peer_addr_param = &peer_addr; |
| uint8_t own_addr_type; |
| bool restart; |
| int rc; |
| |
| rc = parse_arg_init(argc - 1, argv + 1); |
| if (rc != 0) { |
| return rc; |
| } |
| |
| if (argc > 1 && strcmp(argv[1], "stop") == 0) { |
| rc = btshell_adv_stop(); |
| if (rc != 0) { |
| console_printf("advertise stop fail: %d\n", rc); |
| return rc; |
| } |
| |
| return 0; |
| } |
| |
| params.conn_mode = parse_arg_kv_dflt("conn", cmd_adv_conn_modes, |
| BLE_GAP_CONN_MODE_UND, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'conn' parameter\n"); |
| return rc; |
| } |
| |
| params.disc_mode = parse_arg_kv_dflt("discov", cmd_adv_disc_modes, |
| BLE_GAP_DISC_MODE_GEN, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'discov' parameter\n"); |
| return rc; |
| } |
| |
| rc = parse_dev_addr("peer_", cmd_peer_addr_types, &peer_addr); |
| if (rc == ENOENT) { |
| peer_addr_param = NULL; |
| } else if (rc != 0) { |
| console_printf("invalid 'peer_addr' parameter\n"); |
| return rc; |
| } |
| |
| restart = parse_arg_bool_dflt("restart", 0, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'restart' parameter\n"); |
| return rc; |
| } |
| |
| own_addr_type = parse_arg_kv_dflt("own_addr_type", cmd_own_addr_types, |
| btshell_get_default_own_addr_type(), &rc); |
| if (rc != 0) { |
| console_printf("invalid 'own_addr_type' parameter\n"); |
| return rc; |
| } |
| |
| params.channel_map = parse_arg_uint8_dflt("channel_map", 0, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'channel_map' parameter\n"); |
| return rc; |
| } |
| |
| params.filter_policy = parse_arg_kv_dflt("filter", cmd_adv_filt_types, |
| BLE_HCI_ADV_FILT_NONE, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'filter' parameter\n"); |
| return rc; |
| } |
| |
| params.itvl_min = parse_arg_time_dflt("interval_min", 625, 0, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'interval_min' parameter\n"); |
| return rc; |
| } |
| |
| params.itvl_max = parse_arg_time_dflt("interval_max", 625, 0, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'interval_max' parameter\n"); |
| return rc; |
| } |
| |
| params.high_duty_cycle = parse_arg_bool_dflt("high_duty", 0, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'high_duty' parameter\n"); |
| return rc; |
| } |
| |
| duration_ms = parse_arg_long_bounds_dflt("duration", 1, INT32_MAX, |
| BLE_HS_FOREVER, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'duration' parameter\n"); |
| return rc; |
| } |
| |
| rc = btshell_adv_start(own_addr_type, peer_addr_param, duration_ms, |
| ¶ms, restart); |
| if (rc != 0) { |
| console_printf("advertise fail: %d\n", rc); |
| return rc; |
| } |
| |
| return 0; |
| } |
| |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| static const struct shell_param advertise_params[] = { |
| {"stop", "stop advertising procedure"}, |
| {"conn", "connectable mode, usage: =[non|und|dir], default: und"}, |
| {"discov", "discoverable mode, usage: =[non|ltd|gen], default: gen"}, |
| {"peer_addr_type", "usage: =[public|random|public_id|random_id], default: public"}, |
| {"peer_addr", "usage: =[XX:XX:XX:XX:XX:XX]"}, |
| {"own_addr_type", "usage: =[public|random|rpa_pub|rpa_rnd], default: public if available, otherwise random"}, |
| {"channel_map", "usage: =[0x00-0xff], default: 0"}, |
| {"filter", "usage: =[none|scan|conn|both], default: none"}, |
| {"interval_min", "usage: =[0-UINT16_MAX], default: 0"}, |
| {"interval_max", "usage: =[0-UINT16_MAX], default: 0"}, |
| {"high_duty", "usage: =[0-1], default: 0"}, |
| {"duration", "usage: =[1-INT32_MAX], default: INT32_MAX"}, |
| {"restart", "restart advertising after disconnect, usage: =[0-1], default: 0"}, |
| {NULL, NULL} |
| }; |
| |
| static const struct shell_cmd_help advertise_help = { |
| .summary = "start/stop advertising with specific parameters", |
| .usage = NULL, |
| .params = advertise_params, |
| }; |
| #endif |
| #endif |
| |
| /***************************************************************************** |
| * $connect * |
| *****************************************************************************/ |
| |
| static struct parse_arg_kv_pair cmd_ext_conn_phy_opts[] = { |
| { "none", 0x00 }, |
| { "1M", 0x01 }, |
| { "coded", 0x02 }, |
| { "both", 0x03 }, |
| { "all", 0x04 }, |
| { NULL } |
| }; |
| |
| static int |
| cmd_connect(int argc, char **argv) |
| { |
| struct ble_gap_conn_params phy_1M_params = {0}; |
| struct ble_gap_conn_params phy_coded_params = {0}; |
| struct ble_gap_conn_params phy_2M_params = {0}; |
| uint8_t ext; |
| int32_t duration_ms; |
| ble_addr_t peer_addr; |
| ble_addr_t *peer_addr_param = &peer_addr; |
| int own_addr_type; |
| int rc; |
| |
| rc = parse_arg_init(argc - 1, argv + 1); |
| if (rc != 0) { |
| return rc; |
| } |
| |
| if (argc > 1 && strcmp(argv[1], "cancel") == 0) { |
| rc = btshell_conn_cancel(); |
| if (rc != 0) { |
| console_printf("connection cancel fail: %d\n", rc); |
| return rc; |
| } |
| |
| return 0; |
| } |
| |
| ext = parse_arg_kv_dflt("extended", cmd_ext_conn_phy_opts, 0, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'extended' parameter\n"); |
| return rc; |
| } |
| |
| rc = parse_dev_addr("peer_", cmd_peer_addr_types, &peer_addr); |
| if (rc == ENOENT) { |
| /* With no "peer_addr" specified we'll use white list */ |
| peer_addr_param = NULL; |
| } else if (rc != 0) { |
| console_printf("invalid 'peer_addr' parameter\n"); |
| return rc; |
| } |
| |
| own_addr_type = parse_arg_kv_dflt("own_addr_type", cmd_own_addr_types, |
| btshell_get_default_own_addr_type(), &rc); |
| if (rc != 0) { |
| console_printf("invalid 'own_addr_type' parameter\n"); |
| return rc; |
| } |
| |
| duration_ms = parse_arg_long_bounds_dflt("duration", 1, INT32_MAX, 0, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'duration' parameter\n"); |
| return rc; |
| } |
| |
| phy_1M_params.scan_itvl = parse_arg_time_dflt("scan_interval", 625, 0x0010, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'scan_interval' parameter\n"); |
| return rc; |
| } |
| |
| phy_1M_params.scan_window = parse_arg_time_dflt("scan_window", 625, 0x0010, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'scan_window' parameter\n"); |
| return rc; |
| } |
| |
| phy_1M_params.itvl_min = parse_arg_time_dflt("interval_min", 1250, |
| BLE_GAP_INITIAL_CONN_ITVL_MIN, |
| &rc); |
| if (rc != 0) { |
| console_printf("invalid 'interval_min' parameter\n"); |
| return rc; |
| } |
| |
| phy_1M_params.itvl_max = parse_arg_time_dflt("interval_max", 1250, |
| BLE_GAP_INITIAL_CONN_ITVL_MAX, |
| &rc); |
| if (rc != 0) { |
| console_printf("invalid 'interval_max' parameter\n"); |
| return rc; |
| } |
| |
| phy_1M_params.latency = parse_arg_uint16_dflt("latency", 0, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'latency' parameter\n"); |
| return rc; |
| } |
| |
| phy_1M_params.supervision_timeout = parse_arg_time_dflt("timeout", 10000, |
| 0x0100, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'timeout' parameter\n"); |
| return rc; |
| } |
| |
| phy_1M_params.min_ce_len = parse_arg_time_dflt("min_conn_event_len", 625, |
| 0x0010, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'min_conn_event_len' parameter\n"); |
| return rc; |
| } |
| |
| phy_1M_params.max_ce_len = parse_arg_time_dflt("max_conn_event_len", 625, |
| 0x0300, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'max_conn_event_len' parameter\n"); |
| return rc; |
| } |
| |
| if (ext == 0x00) { |
| rc = btshell_conn_initiate(own_addr_type, peer_addr_param, duration_ms, |
| &phy_1M_params); |
| if (rc) { |
| console_printf("error connecting; rc=%d\n", rc); |
| } |
| return rc; |
| } |
| |
| if (ext == 0x01) { |
| rc = btshell_ext_conn_initiate(own_addr_type, peer_addr_param, |
| duration_ms, &phy_1M_params, |
| NULL, NULL); |
| if (rc) { |
| console_printf("error connecting; rc=%d\n", rc); |
| } |
| return rc; |
| } |
| |
| /* Get coded params */ |
| phy_coded_params.scan_itvl = parse_arg_time_dflt("coded_scan_interval", |
| 625, 0x0010, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'coded_scan_interval' parameter\n"); |
| return rc; |
| } |
| |
| phy_coded_params.scan_window = parse_arg_time_dflt("coded_scan_window", |
| 625, 0x0010, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'coded_scan_window' parameter\n"); |
| return rc; |
| } |
| |
| phy_coded_params.itvl_min = parse_arg_time_dflt("coded_interval_min", 1250, |
| BLE_GAP_INITIAL_CONN_ITVL_MIN, |
| &rc); |
| if (rc != 0) { |
| console_printf("invalid 'coded_interval_min' parameter\n"); |
| return rc; |
| } |
| |
| phy_coded_params.itvl_max = parse_arg_time_dflt("coded_interval_max", 1250, |
| BLE_GAP_INITIAL_CONN_ITVL_MAX, |
| &rc); |
| if (rc != 0) { |
| console_printf("invalid 'coded_interval_max' parameter\n"); |
| return rc; |
| } |
| |
| phy_coded_params.latency = |
| parse_arg_uint16_dflt("coded_latency", 0, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'coded_latency' parameter\n"); |
| return rc; |
| } |
| |
| phy_coded_params.supervision_timeout = |
| parse_arg_time_dflt("coded_timeout", 10000, 0x0100, &rc); |
| |
| if (rc != 0) { |
| console_printf("invalid 'coded_timeout' parameter\n"); |
| return rc; |
| } |
| |
| phy_coded_params.min_ce_len = |
| parse_arg_time_dflt("coded_min_conn_event", 625, 0x0010, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'coded_min_conn_event' parameter\n"); |
| return rc; |
| } |
| |
| phy_coded_params.max_ce_len = parse_arg_time_dflt("coded_max_conn_event", |
| 625, 0x0300, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'coded_max_conn_event' parameter\n"); |
| return rc; |
| } |
| |
| /* Get 2M params */ |
| phy_2M_params.itvl_min = parse_arg_time_dflt("2M_interval_min", 1250, |
| BLE_GAP_INITIAL_CONN_ITVL_MIN, |
| &rc); |
| if (rc != 0) { |
| console_printf("invalid '2M_interval_min' parameter\n"); |
| return rc; |
| } |
| |
| phy_2M_params.itvl_max = parse_arg_time_dflt("2M_interval_max", 1250, |
| BLE_GAP_INITIAL_CONN_ITVL_MAX, &rc); |
| if (rc != 0) { |
| console_printf("invalid '2M_interval_max' parameter\n"); |
| return rc; |
| } |
| |
| phy_2M_params.latency = |
| parse_arg_uint16_dflt("2M_latency", 0, &rc); |
| if (rc != 0) { |
| console_printf("invalid '2M_latency' parameter\n"); |
| return rc; |
| } |
| |
| phy_2M_params.supervision_timeout = parse_arg_time_dflt("2M_timeout", 10000, |
| 0x0100, &rc); |
| |
| if (rc != 0) { |
| console_printf("invalid '2M_timeout' parameter\n"); |
| return rc; |
| } |
| |
| phy_2M_params.min_ce_len = parse_arg_time_dflt("2M_min_conn_event", 625, |
| 0x0010, &rc); |
| if (rc != 0) { |
| console_printf("invalid '2M_min_conn_event' parameter\n"); |
| return rc; |
| } |
| |
| phy_2M_params.max_ce_len = parse_arg_time_dflt("2M_max_conn_event", 625, |
| 0x0300, &rc); |
| if (rc != 0) { |
| console_printf("invalid '2M_max_conn_event' parameter\n"); |
| return rc; |
| } |
| |
| if (ext == 0x02) { |
| rc = btshell_ext_conn_initiate(own_addr_type, peer_addr_param, |
| duration_ms, NULL, NULL, &phy_coded_params); |
| return rc; |
| } |
| |
| if (ext == 0x03) { |
| rc = btshell_ext_conn_initiate(own_addr_type, peer_addr_param, |
| duration_ms, &phy_1M_params, NULL, |
| &phy_coded_params); |
| return rc; |
| } |
| |
| rc = btshell_ext_conn_initiate(own_addr_type, peer_addr_param, |
| duration_ms, &phy_1M_params, |
| &phy_2M_params, |
| &phy_coded_params); |
| return rc; |
| } |
| |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| static const struct shell_param connect_params[] = { |
| {"cancel", "cancel connection procedure"}, |
| {"extended", "usage: =[none|1M|coded|both|all], default: none"}, |
| {"peer_addr_type", "usage: =[public|random|public_id|random_id], default: public"}, |
| {"peer_addr", "usage: =[XX:XX:XX:XX:XX:XX]"}, |
| {"own_addr_type", "usage: =[public|random|rpa_pub|rpa_rnd], default: public if available, otherwise random"}, |
| {"duration", "usage: =[1-INT32_MAX], default: 0"}, |
| {"scan_interval", "usage: =[0-UINT16_MAX], default: 0x0010"}, |
| {"scan_window", "usage: =[0-UINT16_MAX], default: 0x0010"}, |
| {"interval_min", "usage: =[0-UINT16_MAX], default: 30"}, |
| {"interval_max", "usage: =[0-UINT16_MAX], default: 50"}, |
| {"latency", "usage: =[UINT16], default: 0"}, |
| {"timeout", "usage: =[UINT16], default: 0x0100"}, |
| {"min_conn_event_len", "usage: =[UINT16], default: 0x0010"}, |
| {"max_conn_event_len", "usage: =[UINT16], default: 0x0300"}, |
| {"coded_scan_interval", "usage: =[0-UINT16_MAX], default: 0x0010"}, |
| {"coded_scan_window", "usage: =[0-UINT16_MAX], default: 0x0010"}, |
| {"coded_interval_min", "usage: =[0-UINT16_MAX], default: 30"}, |
| {"coded_interval_max", "usage: =[0-UINT16_MAX], default: 50"}, |
| {"coded_latency", "usage: =[UINT16], default: 0"}, |
| {"coded_timeout", "usage: =[UINT16], default: 0x0100"}, |
| {"coded_min_conn_event_len", "usage: =[UINT16], default: 0x0010"}, |
| {"coded_max_conn_event_len", "usage: =[UINT16], default: 0x0300"}, |
| {"2M_interval_min", "usage: =[0-UINT16_MAX], default: 30"}, |
| {"2M_interval_max", "usage: =[0-UINT16_MAX], default: 50"}, |
| {"2M_latency", "usage: =[UINT16], default: 0"}, |
| {"2M_timeout", "usage: =[UINT16], default: 0x0100"}, |
| {"2M_min_conn_event_len", "usage: =[UINT16], default: 0x0010"}, |
| {"2M_max_conn_event_len", "usage: =[UINT16], default: 0x0300"}, |
| {NULL, NULL} |
| }; |
| |
| static const struct shell_cmd_help connect_help = { |
| .summary = "start/stop connection procedure with specific parameters", |
| .usage = NULL, |
| .params = connect_params, |
| }; |
| #endif |
| |
| /***************************************************************************** |
| * $disconnect * |
| *****************************************************************************/ |
| |
| static int |
| cmd_disconnect(int argc, char **argv) |
| { |
| uint16_t conn_handle; |
| uint8_t reason; |
| int rc; |
| |
| rc = parse_arg_init(argc - 1, argv + 1); |
| if (rc != 0) { |
| return rc; |
| } |
| |
| conn_handle = parse_arg_uint16("conn", &rc); |
| if (rc != 0) { |
| console_printf("invalid 'conn' parameter\n"); |
| return rc; |
| } |
| |
| reason = parse_arg_uint8_dflt("reason", BLE_ERR_REM_USER_CONN_TERM, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'reason' parameter\n"); |
| return rc; |
| } |
| |
| rc = btshell_term_conn(conn_handle, reason); |
| if (rc != 0) { |
| console_printf("error terminating connection; rc=%d\n", rc); |
| return rc; |
| } |
| |
| return 0; |
| } |
| |
| static int |
| cmd_show_conn(int argc, char **argv) |
| { |
| struct ble_gap_conn_desc conn_desc; |
| struct btshell_conn *conn; |
| int rc; |
| int i; |
| |
| for (i = 0; i < btshell_num_conns; i++) { |
| conn = btshell_conns + i; |
| |
| rc = ble_gap_conn_find(conn->handle, &conn_desc); |
| if (rc == 0) { |
| print_conn_desc(&conn_desc); |
| } |
| } |
| |
| return 0; |
| } |
| |
| static int |
| cmd_show_addr(int argc, char **argv) |
| { |
| uint8_t id_addr[6]; |
| int rc; |
| |
| console_printf("public_id_addr="); |
| rc = ble_hs_id_copy_addr(BLE_ADDR_PUBLIC, id_addr, NULL); |
| if (rc == 0) { |
| print_addr(id_addr); |
| } else { |
| console_printf("none"); |
| } |
| |
| console_printf(" random_id_addr="); |
| rc = ble_hs_id_copy_addr(BLE_ADDR_RANDOM, id_addr, NULL); |
| if (rc == 0) { |
| print_addr(id_addr); |
| } else { |
| console_printf("none"); |
| } |
| console_printf("\n"); |
| |
| return 0; |
| } |
| |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| static const struct shell_param disconnect_params[] = { |
| {"conn", "connection handle parameter, usage: =<UINT16>"}, |
| {"reason", "disconnection reason, usage: =[UINT8], default: 19 (remote user terminated connection)"}, |
| {NULL, NULL} |
| }; |
| |
| static const struct shell_cmd_help disconnect_help = { |
| .summary = "disconnect command", |
| .usage = NULL, |
| .params = disconnect_params, |
| }; |
| #endif |
| |
| /***************************************************************************** |
| * $set-scan-opts * |
| *****************************************************************************/ |
| |
| static struct btshell_scan_opts g_scan_opts = { |
| .limit = UINT16_MAX, |
| .ignore_legacy = 0, |
| .periodic_only = 0, |
| .name_filter_len = 0, |
| }; |
| |
| static int |
| cmd_set_scan_opts(int argc, char **argv) |
| { |
| char *name_filter; |
| int rc; |
| |
| rc = parse_arg_init(argc - 1, argv + 1); |
| if (rc != 0) { |
| return rc; |
| } |
| |
| g_scan_opts.limit = parse_arg_uint16_dflt("decode_limit", UINT16_MAX, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'decode_limit' parameter\n"); |
| return rc; |
| } |
| |
| g_scan_opts.ignore_legacy = parse_arg_bool_dflt("ignore_legacy", 0, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'ignore_legacy' parameter\n"); |
| return rc; |
| } |
| |
| g_scan_opts.periodic_only = parse_arg_bool_dflt("periodic_only", 0, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'periodic_only' parameter\n"); |
| return rc; |
| } |
| |
| name_filter = parse_arg_extract("name_filter"); |
| if (name_filter) { |
| strncpy(g_scan_opts.name_filter, name_filter, NAME_FILTER_LEN_MAX); |
| g_scan_opts.name_filter[NAME_FILTER_LEN_MAX - 1] = '\0'; |
| } else { |
| g_scan_opts.name_filter[0] = '\0'; |
| } |
| |
| g_scan_opts.name_filter_len = strlen(g_scan_opts.name_filter); |
| |
| return rc; |
| } |
| |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| static const struct shell_param set_scan_opts_params[] = { |
| {"decode_limit", "usage: =[0-UINT16_MAX], default: UINT16_MAX"}, |
| {"ignore_legacy", "usage: =[0-1], default: 0"}, |
| {"periodic_only", "usage: =[0-1], default: 0"}, |
| {"name_filter", "usage: =name, default: {none}"}, |
| {NULL, NULL} |
| }; |
| |
| static const struct shell_cmd_help set_scan_opts_help = { |
| .summary = "set scan options", |
| .usage = NULL, |
| .params = set_scan_opts_params, |
| }; |
| #endif |
| |
| /***************************************************************************** |
| * $scan * |
| *****************************************************************************/ |
| |
| static const struct parse_arg_kv_pair cmd_scan_filt_policies[] = { |
| { "no_wl", BLE_HCI_SCAN_FILT_NO_WL }, |
| { "use_wl", BLE_HCI_SCAN_FILT_USE_WL }, |
| { "no_wl_inita", BLE_HCI_SCAN_FILT_NO_WL_INITA }, |
| { "use_wl_inita", BLE_HCI_SCAN_FILT_USE_WL_INITA }, |
| { NULL } |
| }; |
| |
| static struct parse_arg_kv_pair cmd_scan_ext_types[] = { |
| { "none", 0x00 }, |
| { "1M", 0x01 }, |
| { "coded", 0x02 }, |
| { "both", 0x03 }, |
| { NULL } |
| }; |
| |
| static struct btshell_scan_opts g_scan_opts; |
| |
| static int |
| cmd_scan(int argc, char **argv) |
| { |
| struct ble_gap_disc_params params = {0}; |
| struct ble_gap_ext_disc_params uncoded = {0}; |
| struct ble_gap_ext_disc_params coded = {0}; |
| uint8_t extended; |
| int32_t duration_ms; |
| uint8_t own_addr_type; |
| uint16_t duration; |
| uint16_t period; |
| int rc; |
| |
| rc = parse_arg_init(argc - 1, argv + 1); |
| if (rc != 0) { |
| return rc; |
| } |
| |
| if (argc > 1 && (strcmp(argv[1], "cancel") == 0 || strcmp(argv[1], "off") == 0)) { |
| rc = btshell_scan_cancel(); |
| if (rc != 0) { |
| console_printf("scan cancel fail: %d\n", rc); |
| return rc; |
| } |
| return 0; |
| } |
| |
| extended = parse_arg_kv_dflt("extended", cmd_scan_ext_types, 0, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'extended' parameter\n"); |
| return rc; |
| } |
| |
| duration_ms = parse_arg_time_dflt("duration", 10000, BLE_HS_FOREVER, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'duration' parameter\n"); |
| return rc; |
| } |
| |
| params.limited = parse_arg_bool_dflt("limited", 0, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'limited' parameter\n"); |
| return rc; |
| } |
| |
| params.passive = parse_arg_bool_dflt("passive", 0, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'passive' parameter\n"); |
| return rc; |
| } |
| |
| params.itvl = parse_arg_time_dflt("interval", 625, 0, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'interval' parameter\n"); |
| return rc; |
| } |
| |
| params.window = parse_arg_time_dflt("window", 625, 0, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'window' parameter\n"); |
| return rc; |
| } |
| |
| params.filter_policy = parse_arg_kv_dflt("filter", cmd_scan_filt_policies, |
| BLE_HCI_SCAN_FILT_NO_WL, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'filter' parameter\n"); |
| return rc; |
| } |
| |
| params.filter_duplicates = parse_arg_bool_dflt("nodups", 0, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'nodups' parameter\n"); |
| return rc; |
| } |
| |
| own_addr_type = parse_arg_kv_dflt("own_addr_type", cmd_own_addr_types, |
| btshell_get_default_own_addr_type(), &rc); |
| if (rc != 0) { |
| console_printf("invalid 'own_addr_type' parameter\n"); |
| return rc; |
| } |
| |
| if (extended == 0) { |
| rc = btshell_scan(own_addr_type, duration_ms, ¶ms, &g_scan_opts); |
| if (rc != 0) { |
| console_printf("error scanning; rc=%d\n", rc); |
| return rc; |
| } |
| |
| return 0; |
| } |
| |
| /* Copy above parameters to uncoded params */ |
| uncoded.passive = params.passive; |
| uncoded.itvl = params.itvl; |
| uncoded.window = params.window; |
| |
| duration = parse_arg_time_dflt("extended_duration", 10000, 0, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'extended_duration' parameter\n"); |
| return rc; |
| } |
| |
| period = parse_arg_time_dflt("extended_period", 1280000, 0, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'extended_period' parameter\n"); |
| return rc; |
| } |
| |
| coded.itvl = parse_arg_time_dflt("longrange_interval", 625, 0, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'longrange_interval' parameter\n"); |
| return rc; |
| } |
| |
| coded.window = parse_arg_time_dflt("longrange_window", 625, 0, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'longrange_window' parameter\n"); |
| return rc; |
| } |
| |
| coded.passive = parse_arg_uint16_dflt("longrange_passive", 0, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'longrange_passive' parameter\n"); |
| return rc; |
| } |
| |
| switch (extended) { |
| case 0x01: |
| rc = btshell_ext_scan(own_addr_type, duration, period, |
| params.filter_duplicates, params.filter_policy, |
| params.limited, &uncoded, NULL, |
| &g_scan_opts); |
| break; |
| case 0x02: |
| rc = btshell_ext_scan(own_addr_type, duration, period, |
| params.filter_duplicates, params.filter_policy, |
| params.limited, NULL, &coded, |
| &g_scan_opts); |
| break; |
| case 0x03: |
| rc = btshell_ext_scan(own_addr_type, duration, period, |
| params.filter_duplicates, params.filter_policy, |
| params.limited, &uncoded, &coded, |
| &g_scan_opts); |
| break; |
| default: |
| assert(0); |
| break; |
| } |
| |
| if (rc != 0) { |
| console_printf("error scanning; rc=%d\n", rc); |
| } |
| |
| return rc; |
| } |
| |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| static const struct shell_param scan_params[] = { |
| {"cancel", "cancel scan procedure"}, |
| {"off", "\"cancel\" param substitute"}, |
| {"extended", "usage: =[none|1M|coded|both], default: none"}, |
| {"duration", "usage: =[1-INT32_MAX], default: INT32_MAX"}, |
| {"limited", "usage: =[0-1], default: 0"}, |
| {"passive", "usage: =[0-1], default: 0"}, |
| {"interval", "usage: =[0-UINT16_MAX], default: 0"}, |
| {"window", "usage: =[0-UINT16_MAX], default: 0"}, |
| {"filter", "usage: =[no_wl|use_wl|no_wl_inita|use_wl_inita], default: no_wl"}, |
| {"nodups", "usage: =[0-1], default: 0"}, |
| {"own_addr_type", "usage: =[public|random|rpa_pub|rpa_rnd], default: public if available, otherwise random"}, |
| {"extended_duration", "usage: =[0-UINT16_MAX], default: 0"}, |
| {"extended_period", "usage: =[0-UINT16_MAX], default: 0"}, |
| {"longrange_interval", "usage: =[0-UINT16_MAX], default: 0"}, |
| {"longrange_window", "usage: =[0-UINT16_MAX], default: 0"}, |
| {"longrange_passive", "usage: =[0-1], default: 0"}, |
| {NULL, NULL} |
| }; |
| |
| static const struct shell_cmd_help scan_help = { |
| .summary = "start/stop scan procedure with specific parameters", |
| .usage = NULL, |
| .params = scan_params, |
| }; |
| #endif |
| |
| /***************************************************************************** |
| * $set * |
| *****************************************************************************/ |
| |
| static int |
| cmd_set_addr(void) |
| { |
| ble_addr_t addr; |
| int rc; |
| |
| rc = parse_dev_addr(NULL, cmd_addr_type, &addr); |
| if (rc != 0) { |
| console_printf("invalid 'addr' parameter\n"); |
| return rc; |
| } |
| |
| switch (addr.type) { |
| #if MYNEWT_VAL(BLE_CONTROLLER) |
| case BLE_ADDR_PUBLIC: |
| /* We shouldn't be writing to the controller's address (g_dev_addr). |
| * There is no standard way to set the local public address, so this is |
| * our only option at the moment. |
| */ |
| memcpy(g_dev_addr, addr.val, 6); |
| ble_hs_id_set_pub(g_dev_addr); |
| break; |
| #endif |
| |
| case BLE_ADDR_RANDOM: |
| rc = ble_hs_id_set_rnd(addr.val); |
| if (rc != 0) { |
| return rc; |
| } |
| break; |
| |
| default: |
| return BLE_HS_EUNKNOWN; |
| } |
| |
| return 0; |
| } |
| |
| static int |
| cmd_set(int argc, char **argv) |
| { |
| uint16_t mtu; |
| uint8_t irk[16]; |
| int good = 0; |
| int rc; |
| |
| rc = parse_arg_init(argc - 1, argv + 1); |
| if (rc != 0) { |
| return rc; |
| } |
| |
| rc = parse_arg_find_idx("addr"); |
| if (rc != -1) { |
| rc = cmd_set_addr(); |
| if (rc != 0) { |
| return rc; |
| } |
| good = 1; |
| } |
| |
| mtu = parse_arg_uint16("mtu", &rc); |
| if (rc == 0) { |
| rc = ble_att_set_preferred_mtu(mtu); |
| if (rc == 0) { |
| good = 1; |
| } |
| } else if (rc != ENOENT) { |
| console_printf("invalid 'mtu' parameter\n"); |
| return rc; |
| } |
| |
| rc = parse_arg_byte_stream_exact_length("irk", irk, 16); |
| if (rc == 0) { |
| good = 1; |
| ble_hs_pvcy_set_our_irk(irk); |
| } else if (rc != ENOENT) { |
| console_printf("invalid 'irk' parameter\n"); |
| return rc; |
| } |
| |
| if (!good) { |
| console_printf("Error: no valid settings specified\n"); |
| return -1; |
| } |
| |
| return 0; |
| } |
| |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| static const struct shell_param set_params[] = { |
| {"addr", "set device address, usage: =[XX:XX:XX:XX:XX:XX]"}, |
| {"addr_type", "set device address type, usage: =[public|random], default: public"}, |
| {"mtu", "Maximum Transimssion Unit, usage: =[0-UINT16_MAX]"}, |
| {"irk", "Identity Resolving Key, usage: =[XX:XX...], len=16 octets"}, |
| {NULL, NULL} |
| }; |
| |
| static const struct shell_cmd_help set_help = { |
| .summary = "set device parameters", |
| .usage = NULL, |
| .params = set_params, |
| }; |
| #endif |
| |
| /***************************************************************************** |
| * $set-adv-data * |
| *****************************************************************************/ |
| |
| #define CMD_ADV_DATA_MAX_UUIDS16 8 |
| #define CMD_ADV_DATA_MAX_UUIDS32 8 |
| #define CMD_ADV_DATA_MAX_UUIDS128 2 |
| #define CMD_ADV_DATA_MAX_PUBLIC_TGT_ADDRS 8 |
| #define CMD_ADV_DATA_SVC_DATA_UUID16_MAX_LEN BLE_HS_ADV_MAX_FIELD_SZ |
| #define CMD_ADV_DATA_SVC_DATA_UUID32_MAX_LEN BLE_HS_ADV_MAX_FIELD_SZ |
| #define CMD_ADV_DATA_SVC_DATA_UUID128_MAX_LEN BLE_HS_ADV_MAX_FIELD_SZ |
| #define CMD_ADV_DATA_URI_MAX_LEN BLE_HS_ADV_MAX_FIELD_SZ |
| #define CMD_ADV_DATA_MFG_DATA_MAX_LEN BLE_HS_ADV_MAX_FIELD_SZ |
| |
| #if MYNEWT_VAL(BLE_EXT_ADV) |
| static void |
| update_pattern(uint8_t *buf, int counter) |
| { |
| int i; |
| |
| for (i = 0; i < 10; i += 2) { |
| counter += 2; |
| buf[i] = (counter / 1000) << 4 | (counter / 100 % 10); |
| buf[i + 1] = (counter / 10 % 10) << 4 | (counter % 10); |
| } |
| } |
| #endif |
| |
| static int |
| cmd_set_adv_data_or_scan_rsp(int argc, char **argv, bool scan_rsp, |
| bool periodic) |
| { |
| static bssnz_t ble_uuid16_t uuids16[CMD_ADV_DATA_MAX_UUIDS16]; |
| static bssnz_t ble_uuid32_t uuids32[CMD_ADV_DATA_MAX_UUIDS32]; |
| static bssnz_t ble_uuid128_t uuids128[CMD_ADV_DATA_MAX_UUIDS128]; |
| static bssnz_t uint8_t |
| public_tgt_addrs[CMD_ADV_DATA_MAX_PUBLIC_TGT_ADDRS] |
| [BLE_HS_ADV_PUBLIC_TGT_ADDR_ENTRY_LEN]; |
| static bssnz_t uint8_t slave_itvl_range[BLE_HS_ADV_SLAVE_ITVL_RANGE_LEN]; |
| static bssnz_t uint8_t |
| svc_data_uuid16[CMD_ADV_DATA_SVC_DATA_UUID16_MAX_LEN]; |
| static bssnz_t uint8_t |
| svc_data_uuid32[CMD_ADV_DATA_SVC_DATA_UUID32_MAX_LEN]; |
| static bssnz_t uint8_t |
| svc_data_uuid128[CMD_ADV_DATA_SVC_DATA_UUID128_MAX_LEN]; |
| static bssnz_t uint8_t uri[CMD_ADV_DATA_URI_MAX_LEN]; |
| static bssnz_t uint8_t mfg_data[CMD_ADV_DATA_MFG_DATA_MAX_LEN]; |
| struct ble_hs_adv_fields adv_fields; |
| uint32_t uuid32; |
| uint16_t uuid16; |
| uint8_t uuid128[16]; |
| uint8_t public_tgt_addr[BLE_HS_ADV_PUBLIC_TGT_ADDR_ENTRY_LEN]; |
| uint8_t eddystone_url_body_len; |
| uint8_t eddystone_url_suffix; |
| uint8_t eddystone_url_scheme; |
| int8_t eddystone_measured_power = 0; |
| char eddystone_url_body[BLE_EDDYSTONE_URL_MAX_LEN]; |
| char *eddystone_url_full; |
| unsigned int svc_data_uuid16_len; |
| unsigned int svc_data_uuid32_len; |
| unsigned int svc_data_uuid128_len; |
| unsigned int uri_len; |
| unsigned int mfg_data_len; |
| int tmp; |
| int rc; |
| #if MYNEWT_VAL(BLE_EXT_ADV) |
| uint8_t instance; |
| uint8_t extra_data[10]; |
| uint16_t counter; |
| uint16_t extra_data_len; |
| struct os_mbuf *adv_data; |
| #endif |
| |
| /* cannot set scan rsp for periodic */ |
| if (scan_rsp && periodic) { |
| return -1; |
| } |
| |
| memset(&adv_fields, 0, sizeof adv_fields); |
| |
| rc = parse_arg_init(argc - 1, argv + 1); |
| if (rc != 0) { |
| return rc; |
| } |
| |
| #if MYNEWT_VAL(BLE_EXT_ADV) |
| instance = parse_arg_uint8_dflt("instance", 0, &rc); |
| if (rc != 0 || instance >= BLE_ADV_INSTANCES) { |
| console_printf("invalid instance\n"); |
| return rc; |
| } |
| #endif |
| |
| tmp = parse_arg_uint8("flags", &rc); |
| if (rc == 0) { |
| adv_fields.flags = tmp; |
| } else if (rc != ENOENT) { |
| console_printf("invalid 'flags' parameter\n"); |
| return rc; |
| } |
| |
| while (1) { |
| uuid16 = parse_arg_uint16("uuid16", &rc); |
| if (rc == 0) { |
| if (adv_fields.num_uuids16 >= CMD_ADV_DATA_MAX_UUIDS16) { |
| console_printf("invalid 'uuid16' parameter\n"); |
| return EINVAL; |
| } |
| uuids16[adv_fields.num_uuids16] = (ble_uuid16_t) BLE_UUID16_INIT(uuid16); |
| adv_fields.num_uuids16++; |
| } else if (rc == ENOENT) { |
| break; |
| } else { |
| console_printf("invalid 'uuid16' parameter\n"); |
| return rc; |
| } |
| } |
| if (adv_fields.num_uuids16 > 0) { |
| adv_fields.uuids16 = uuids16; |
| } |
| |
| tmp = parse_arg_bool_dflt("uuids16_is_complete", 0, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'uuids16_is_complete' parameter\n"); |
| return rc; |
| } |
| |
| while (1) { |
| uuid32 = parse_arg_uint32("uuid32", &rc); |
| if (rc == 0) { |
| if (adv_fields.num_uuids32 >= CMD_ADV_DATA_MAX_UUIDS32) { |
| console_printf("invalid 'uuid32' parameter\n"); |
| return EINVAL; |
| } |
| uuids32[adv_fields.num_uuids32] = (ble_uuid32_t) BLE_UUID32_INIT(uuid32); |
| adv_fields.num_uuids32++; |
| } else if (rc == ENOENT) { |
| break; |
| } else { |
| console_printf("invalid 'uuid32' parameter\n"); |
| return rc; |
| } |
| } |
| if (adv_fields.num_uuids32 > 0) { |
| adv_fields.uuids32 = uuids32; |
| } |
| |
| tmp = parse_arg_bool_dflt("uuids32_is_complete", 0, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'uuids32_is_complete' parameter\n"); |
| return rc; |
| } |
| |
| while (1) { |
| rc = parse_arg_byte_stream_exact_length("uuid128", uuid128, 16); |
| if (rc == 0) { |
| if (adv_fields.num_uuids128 >= CMD_ADV_DATA_MAX_UUIDS128) { |
| console_printf("invalid 'uuid128' parameter\n"); |
| return EINVAL; |
| } |
| ble_uuid_init_from_buf((ble_uuid_any_t *) &uuids128[adv_fields.num_uuids128], |
| uuid128, 16); |
| adv_fields.num_uuids128++; |
| } else if (rc == ENOENT) { |
| break; |
| } else { |
| console_printf("invalid 'uuid128' parameter\n"); |
| return rc; |
| } |
| } |
| if (adv_fields.num_uuids128 > 0) { |
| adv_fields.uuids128 = uuids128; |
| } |
| |
| tmp = parse_arg_bool_dflt("uuids128_is_complete", 0, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'uuids128_is_complete' parameter\n"); |
| return rc; |
| } |
| |
| adv_fields.name = (uint8_t *)parse_arg_extract("name"); |
| if (adv_fields.name != NULL) { |
| adv_fields.name_len = strlen((char *)adv_fields.name); |
| } |
| |
| tmp = parse_arg_long_bounds("tx_power_level", INT8_MIN, INT8_MAX, &rc); |
| if (rc == 0) { |
| adv_fields.tx_pwr_lvl = tmp; |
| adv_fields.tx_pwr_lvl_is_present = 1; |
| } else if (rc != ENOENT) { |
| console_printf("invalid 'tx_power_level' parameter\n"); |
| return rc; |
| } |
| |
| rc = parse_arg_byte_stream_exact_length("slave_interval_range", |
| slave_itvl_range, |
| BLE_HS_ADV_SLAVE_ITVL_RANGE_LEN); |
| if (rc == 0) { |
| adv_fields.slave_itvl_range = slave_itvl_range; |
| } else if (rc != ENOENT) { |
| console_printf("invalid 'slave_interval_range' parameter\n"); |
| return rc; |
| } |
| |
| rc = parse_arg_byte_stream("service_data_uuid16", |
| CMD_ADV_DATA_SVC_DATA_UUID16_MAX_LEN, |
| svc_data_uuid16, &svc_data_uuid16_len); |
| if (rc == 0) { |
| adv_fields.svc_data_uuid16 = svc_data_uuid16; |
| adv_fields.svc_data_uuid16_len = svc_data_uuid16_len; |
| } else if (rc != ENOENT) { |
| console_printf("invalid 'service_data_uuid16' parameter\n"); |
| return rc; |
| } |
| |
| while (1) { |
| rc = parse_arg_byte_stream_exact_length( |
| "public_target_address", public_tgt_addr, |
| BLE_HS_ADV_PUBLIC_TGT_ADDR_ENTRY_LEN); |
| if (rc == 0) { |
| if (adv_fields.num_public_tgt_addrs >= |
| CMD_ADV_DATA_MAX_PUBLIC_TGT_ADDRS) { |
| |
| console_printf("invalid 'public_target_address' parameter\n"); |
| return EINVAL; |
| } |
| memcpy(public_tgt_addrs[adv_fields.num_public_tgt_addrs], |
| public_tgt_addr, BLE_HS_ADV_PUBLIC_TGT_ADDR_ENTRY_LEN); |
| adv_fields.num_public_tgt_addrs++; |
| } else if (rc == ENOENT) { |
| break; |
| } else { |
| console_printf("invalid 'public_target_address' parameter\n"); |
| return rc; |
| } |
| } |
| if (adv_fields.num_public_tgt_addrs > 0) { |
| adv_fields.public_tgt_addr = (void *)public_tgt_addrs; |
| } |
| |
| adv_fields.appearance = parse_arg_uint16("appearance", &rc); |
| if (rc == 0) { |
| adv_fields.appearance_is_present = 1; |
| } else if (rc != ENOENT) { |
| console_printf("invalid 'appearance' parameter\n"); |
| return rc; |
| } |
| |
| adv_fields.adv_itvl = parse_arg_uint16("advertising_interval", &rc); |
| if (rc == 0) { |
| adv_fields.adv_itvl_is_present = 1; |
| } else if (rc != ENOENT) { |
| console_printf("invalid 'advertising_interval' parameter\n"); |
| return rc; |
| } |
| |
| rc = parse_arg_byte_stream("service_data_uuid32", |
| CMD_ADV_DATA_SVC_DATA_UUID32_MAX_LEN, |
| svc_data_uuid32, &svc_data_uuid32_len); |
| if (rc == 0) { |
| adv_fields.svc_data_uuid32 = svc_data_uuid32; |
| adv_fields.svc_data_uuid32_len = svc_data_uuid32_len; |
| } else if (rc != ENOENT) { |
| console_printf("invalid 'service_data_uuid32' parameter\n"); |
| return rc; |
| } |
| |
| rc = parse_arg_byte_stream_custom("service_data_uuid128", ":-", |
| CMD_ADV_DATA_SVC_DATA_UUID128_MAX_LEN, |
| svc_data_uuid128, 0, &svc_data_uuid128_len); |
| if (rc == 0) { |
| adv_fields.svc_data_uuid128 = svc_data_uuid128; |
| adv_fields.svc_data_uuid128_len = svc_data_uuid128_len; |
| } else if (rc != ENOENT) { |
| console_printf("invalid 'service_data_uuid128' parameter\n"); |
| return rc; |
| } |
| |
| rc = parse_arg_byte_stream("uri", CMD_ADV_DATA_URI_MAX_LEN, uri, &uri_len); |
| if (rc == 0) { |
| adv_fields.uri = uri; |
| adv_fields.uri_len = uri_len; |
| } else if (rc != ENOENT) { |
| console_printf("invalid 'uri' parameter\n"); |
| return rc; |
| } |
| |
| rc = parse_arg_byte_stream("mfg_data", CMD_ADV_DATA_MFG_DATA_MAX_LEN, |
| mfg_data, &mfg_data_len); |
| if (rc == 0) { |
| adv_fields.mfg_data = mfg_data; |
| adv_fields.mfg_data_len = mfg_data_len; |
| } else if (rc != ENOENT) { |
| console_printf("invalid 'mfg_data' parameter\n"); |
| return rc; |
| } |
| |
| tmp = parse_arg_long_bounds("eddystone_measured_power", -100, 20, &rc); |
| if (rc == 0) { |
| eddystone_measured_power = tmp; |
| } else if (rc != ENOENT) { |
| console_printf("invalid 'eddystone_measured_power' parameter\n"); |
| return rc; |
| } |
| |
| eddystone_url_full = parse_arg_extract("eddystone_url"); |
| if (eddystone_url_full != NULL) { |
| rc = parse_eddystone_url(eddystone_url_full, &eddystone_url_scheme, |
| eddystone_url_body, |
| &eddystone_url_body_len, |
| &eddystone_url_suffix); |
| if (rc != 0) { |
| goto done; |
| } |
| |
| rc = ble_eddystone_set_adv_data_url(&adv_fields, eddystone_url_scheme, |
| eddystone_url_body, |
| eddystone_url_body_len, |
| eddystone_url_suffix, |
| eddystone_measured_power); |
| } else { |
| #if MYNEWT_VAL(BLE_EXT_ADV) |
| /* Default to legacy PDUs size, mbuf chain will be increased if needed |
| */ |
| adv_data = os_msys_get_pkthdr(BLE_HCI_MAX_ADV_DATA_LEN, 0); |
| if (!adv_data) { |
| rc = ENOMEM; |
| goto done; |
| } |
| |
| rc = ble_hs_adv_set_fields_mbuf(&adv_fields, adv_data); |
| if (rc) { |
| os_mbuf_free_chain(adv_data); |
| goto done; |
| } |
| |
| /* Append some extra data, if requested */ |
| extra_data_len = parse_arg_uint16("extra_data_len", &rc); |
| if (rc == 0) { |
| counter = 0; |
| extra_data_len = min(extra_data_len, 1650); |
| while (counter < extra_data_len) { |
| update_pattern(extra_data, counter); |
| |
| rc = os_mbuf_append(adv_data, extra_data, |
| min(extra_data_len - counter, 10)); |
| if (rc) { |
| os_mbuf_free_chain(adv_data); |
| goto done; |
| } |
| |
| counter += 10; |
| } |
| } |
| |
| if (scan_rsp) { |
| rc = ble_gap_ext_adv_rsp_set_data(instance, adv_data); |
| #if MYNEWT_VAL(BLE_PERIODIC_ADV) |
| } else if (periodic) { |
| rc = ble_gap_periodic_adv_set_data(instance, adv_data, NULL); |
| #endif |
| } else { |
| rc = ble_gap_ext_adv_set_data(instance, adv_data); |
| } |
| #else |
| if (scan_rsp) { |
| rc = ble_gap_adv_rsp_set_fields(&adv_fields); |
| } else { |
| rc = ble_gap_adv_set_fields(&adv_fields); |
| } |
| #endif |
| } |
| done: |
| if (rc != 0) { |
| console_printf("error setting advertisement data; rc=%d\n", rc); |
| return rc; |
| } |
| |
| return 0; |
| } |
| |
| static int |
| cmd_set_adv_data(int argc, char **argv) |
| { |
| return cmd_set_adv_data_or_scan_rsp(argc, argv, false, false); |
| } |
| |
| static int |
| cmd_set_scan_rsp(int argc, char **argv) |
| { |
| return cmd_set_adv_data_or_scan_rsp(argc, argv, true, false); |
| } |
| |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| static const struct shell_param set_adv_data_params[] = { |
| {"instance", "default: 0"}, |
| {"flags", "usage: =[0-UINT8_MAX]"}, |
| {"uuid16", "usage: =[UINT16]"}, |
| {"uuid16_is_complete", "usage: =[0-1], default=0"}, |
| {"uuid32", "usage: =[UINT32]"}, |
| {"uuid32_is_complete", "usage: =[0-1], default=0"}, |
| {"uuid128", "usage: =[XX:XX...], len=16 octets"}, |
| {"uuid128_is_complete", "usage: =[0-1], default=0"}, |
| {"tx_power_level", "usage: =[INT8_MIN-INT8_MAX]"}, |
| {"slave_interval_range", "usage: =[XX:XX:XX:XX]"}, |
| {"public_target_address", "usage: =[XX:XX:XX:XX:XX:XX]"}, |
| {"appearance", "usage: =[UINT16]"}, |
| {"name", "usage: =[string]"}, |
| {"advertising_interval", "usage: =[UINT16]"}, |
| {"service_data_uuid16", "usage: =[XX:XX...]"}, |
| {"service_data_uuid32", "usage: =[XX:XX...]"}, |
| {"service_data_uuid128", "usage: =[XX:XX...]"}, |
| {"uri", "usage: =[XX:XX...]"}, |
| {"mfg_data", "usage: =[XX:XX...]"}, |
| {"measured_power", "usage: =[-100-20]"}, |
| {"eddystone_url", "usage: =[string]"}, |
| #if MYNEWT_VAL(BLE_EXT_ADV) |
| {"extra_data_len", "usage: =[UINT16]"}, |
| #endif |
| {NULL, NULL} |
| }; |
| |
| static const struct shell_cmd_help set_adv_data_help = { |
| .summary = "set advertising data", |
| .usage = NULL, |
| .params = set_adv_data_params, |
| }; |
| |
| static const struct shell_cmd_help set_scan_rsp_help = { |
| .summary = "set scan response", |
| .usage = NULL, |
| .params = set_adv_data_params, |
| }; |
| #endif |
| |
| /***************************************************************************** |
| * $set-priv-mode * |
| *****************************************************************************/ |
| |
| static int |
| cmd_set_priv_mode(int argc, char **argv) |
| { |
| ble_addr_t addr; |
| uint8_t priv_mode; |
| int rc; |
| |
| rc = parse_arg_init(argc - 1, argv + 1); |
| if (rc != 0) { |
| return rc; |
| } |
| |
| rc = parse_dev_addr(NULL, cmd_addr_type, &addr); |
| if (rc != 0) { |
| console_printf("invalid 'addr' parameter\n"); |
| return rc; |
| } |
| |
| priv_mode = parse_arg_uint8("mode", &rc); |
| if (rc != 0) { |
| console_printf("missing mode\n"); |
| return rc; |
| } |
| |
| return ble_gap_set_priv_mode(&addr, priv_mode); |
| } |
| |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| static const struct shell_param set_priv_mode_params[] = { |
| {"addr", "set priv mode for device address, usage: =[XX:XX:XX:XX:XX:XX]"}, |
| {"addr_type", "set priv mode for device address type, usage: =[public|random], default: public"}, |
| {"mode", "set priv mode, usage: =[0-UINT8_MAX]"}, |
| {NULL, NULL} |
| }; |
| |
| static const struct shell_cmd_help set_priv_mode_help = { |
| .summary = "set priv mode", |
| .usage = NULL, |
| .params = set_priv_mode_params, |
| }; |
| #endif |
| |
| /***************************************************************************** |
| * $white-list * |
| *****************************************************************************/ |
| |
| #define CMD_WL_MAX_SZ 8 |
| |
| static int |
| cmd_white_list(int argc, char **argv) |
| { |
| static ble_addr_t addrs[CMD_WL_MAX_SZ]; |
| int addrs_cnt; |
| int rc; |
| |
| rc = parse_arg_init(argc - 1, argv + 1); |
| if (rc != 0) { |
| return rc; |
| } |
| |
| addrs_cnt = 0; |
| while (1) { |
| if (addrs_cnt >= CMD_WL_MAX_SZ) { |
| return EINVAL; |
| } |
| |
| rc = parse_dev_addr(NULL, cmd_addr_type, &addrs[addrs_cnt]); |
| if (rc == ENOENT) { |
| break; |
| } else if (rc != 0) { |
| console_printf("invalid 'addr' parameter #%d\n", addrs_cnt + 1); |
| return rc; |
| } |
| |
| addrs_cnt++; |
| } |
| |
| if (addrs_cnt == 0) { |
| return EINVAL; |
| } |
| |
| btshell_wl_set(addrs, addrs_cnt); |
| |
| return 0; |
| } |
| |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| static const struct shell_param white_list_params[] = { |
| {"addr", "white-list device addresses, usage: =[XX:XX:XX:XX:XX:XX]"}, |
| {"addr_type", "white-list address types, usage: =[public|random]"}, |
| {NULL, NULL} |
| }; |
| |
| static const struct shell_cmd_help white_list_help = { |
| .summary = "set white-list addresses", |
| .usage = NULL, |
| .params = white_list_params, |
| }; |
| #endif |
| |
| /***************************************************************************** |
| * $conn-rssi * |
| *****************************************************************************/ |
| |
| static int |
| cmd_conn_rssi(int argc, char **argv) |
| { |
| uint16_t conn_handle; |
| int8_t rssi; |
| int rc; |
| |
| rc = parse_arg_init(argc - 1, argv + 1); |
| if (rc != 0) { |
| return rc; |
| } |
| |
| conn_handle = parse_arg_uint16("conn", &rc); |
| if (rc != 0) { |
| console_printf("invalid 'conn' parameter\n"); |
| return rc; |
| } |
| |
| rc = btshell_rssi(conn_handle, &rssi); |
| if (rc != 0) { |
| console_printf("error reading rssi; rc=%d\n", rc); |
| return rc; |
| } |
| |
| console_printf("conn=%d rssi=%d\n", conn_handle, rssi); |
| |
| return 0; |
| } |
| |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| static const struct shell_param conn_rssi_params[] = { |
| {"conn", "connection handle parameter, usage: =<UINT16>"}, |
| {NULL, NULL} |
| }; |
| |
| static const struct shell_cmd_help conn_rssi_help = { |
| .summary = "check connection rssi", |
| .usage = NULL, |
| .params = conn_rssi_params, |
| }; |
| #endif |
| |
| /***************************************************************************** |
| * $conn-update-params * |
| *****************************************************************************/ |
| |
| static int |
| cmd_conn_update_params(int argc, char **argv) |
| { |
| struct ble_gap_upd_params params; |
| uint16_t conn_handle; |
| int rc; |
| |
| rc = parse_arg_init(argc - 1, argv + 1); |
| if (rc != 0) { |
| return rc; |
| } |
| |
| conn_handle = parse_arg_uint16("conn", &rc); |
| if (rc != 0) { |
| console_printf("invalid 'conn' parameter\n"); |
| return rc; |
| } |
| |
| params.itvl_min = parse_arg_time_dflt("interval_min", 1250, |
| BLE_GAP_INITIAL_CONN_ITVL_MIN, |
| &rc); |
| if (rc != 0) { |
| console_printf("invalid 'interval_min' parameter\n"); |
| return rc; |
| } |
| |
| params.itvl_max = parse_arg_time_dflt("interval_max", 1250, |
| BLE_GAP_INITIAL_CONN_ITVL_MAX, |
| &rc); |
| if (rc != 0) { |
| console_printf("invalid 'interval_max' parameter\n"); |
| return rc; |
| } |
| |
| params.latency = parse_arg_uint16_dflt("latency", 0, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'latency' parameter\n"); |
| return rc; |
| } |
| |
| params.supervision_timeout = parse_arg_time_dflt("timeout", 10000, 0x0100, |
| &rc); |
| if (rc != 0) { |
| console_printf("invalid 'timeout' parameter\n"); |
| return rc; |
| } |
| |
| params.min_ce_len = parse_arg_time_dflt("min_conn_event_len", 625, |
| 0x0010, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'min_conn_event_len' parameter\n"); |
| return rc; |
| } |
| |
| params.max_ce_len = parse_arg_time_dflt("max_conn_event_len", 625, |
| 0x0300, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'max_conn_event_len' parameter\n"); |
| return rc; |
| } |
| |
| rc = btshell_update_conn(conn_handle, ¶ms); |
| if (rc != 0) { |
| console_printf("error updating connection; rc=%d\n", rc); |
| return rc; |
| } |
| |
| return 0; |
| } |
| |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| static const struct shell_param conn_update_params_params[] = { |
| {"conn", "conn_update_paramsion handle, usage: =<UINT16>"}, |
| {"interval_min", "usage: =[0-UINT16_MAX], default: 30"}, |
| {"interval_max", "usage: =[0-UINT16_MAX], default: 50"}, |
| {"latency", "usage: =[UINT16], default: 0"}, |
| {"timeout", "usage: =[UINT16], default: 0x0100"}, |
| {"min_conn_event_len", "usage: =[UINT16], default: 0x0010"}, |
| {"max_conn_event_len", "usage: =[UINT16], default: 0x0300"}, |
| {NULL, NULL} |
| }; |
| |
| static const struct shell_cmd_help conn_update_params_help = { |
| .summary = "update connection parameters", |
| .usage = "conn_update_params usage", |
| .params = conn_update_params_params, |
| }; |
| #endif |
| |
| /***************************************************************************** |
| * $conn-datalen * |
| *****************************************************************************/ |
| |
| static int |
| cmd_conn_datalen(int argc, char **argv) |
| { |
| uint16_t conn_handle; |
| uint16_t tx_octets; |
| uint16_t tx_time; |
| int rc; |
| |
| rc = parse_arg_init(argc - 1, argv + 1); |
| if (rc != 0) { |
| return rc; |
| } |
| |
| conn_handle = parse_arg_uint16("conn", &rc); |
| if (rc != 0) { |
| console_printf("invalid 'conn' parameter\n"); |
| return rc; |
| } |
| |
| tx_octets = parse_arg_uint16("octets", &rc); |
| if (rc != 0) { |
| console_printf("invalid 'octets' parameter\n"); |
| return rc; |
| } |
| |
| tx_time = parse_arg_uint16("time", &rc); |
| if (rc != 0) { |
| console_printf("invalid 'time' parameter\n"); |
| return rc; |
| } |
| |
| rc = btshell_datalen(conn_handle, tx_octets, tx_time); |
| if (rc != 0) { |
| console_printf("error setting data length; rc=%d\n", rc); |
| return rc; |
| } |
| |
| return 0; |
| } |
| |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| static const struct shell_param conn_datalen_params[] = { |
| {"conn", "Connection handle, usage: =<UINT16>"}, |
| {"octets", "Max payload size to include in LL Data PDU, " |
| "range=<27-251>, usage: =<UINT16>"}, |
| {"time", "Max number of microseconds the controller should use to tx " |
| "single LL packet, range=<328-17040>, usage: =<UINT16>"}, |
| {NULL, NULL} |
| }; |
| |
| static const struct shell_cmd_help conn_datalen_help = { |
| .summary = "set data length parameters for connection", |
| .usage = NULL, |
| .params = conn_datalen_params, |
| }; |
| #endif |
| |
| /***************************************************************************** |
| * keystore * |
| *****************************************************************************/ |
| |
| static const struct parse_arg_kv_pair cmd_keystore_entry_type[] = { |
| { "msec", BLE_STORE_OBJ_TYPE_PEER_SEC }, |
| { "ssec", BLE_STORE_OBJ_TYPE_OUR_SEC }, |
| { "cccd", BLE_STORE_OBJ_TYPE_CCCD }, |
| { NULL } |
| }; |
| |
| static int |
| cmd_keystore_parse_keydata(int argc, char **argv, union ble_store_key *out, |
| int *obj_type) |
| { |
| int rc; |
| |
| memset(out, 0, sizeof(*out)); |
| *obj_type = parse_arg_kv("type", cmd_keystore_entry_type, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'type' parameter\n"); |
| return rc; |
| } |
| |
| switch (*obj_type) { |
| case BLE_STORE_OBJ_TYPE_PEER_SEC: |
| case BLE_STORE_OBJ_TYPE_OUR_SEC: |
| rc = parse_dev_addr(NULL, cmd_addr_type, &out->sec.peer_addr); |
| if (rc != 0) { |
| console_printf("invalid 'addr' parameter\n"); |
| return rc; |
| } |
| |
| return 0; |
| |
| default: |
| return EINVAL; |
| } |
| } |
| |
| static int |
| cmd_keystore_parse_valuedata(int argc, char **argv, |
| int obj_type, |
| union ble_store_key *key, |
| union ble_store_value *out) |
| { |
| int rc; |
| int valcnt = 0; |
| memset(out, 0, sizeof(*out)); |
| |
| switch (obj_type) { |
| case BLE_STORE_OBJ_TYPE_PEER_SEC: |
| case BLE_STORE_OBJ_TYPE_OUR_SEC: |
| rc = parse_arg_byte_stream_exact_length("ltk", out->sec.ltk, 16); |
| if (rc == 0) { |
| out->sec.ltk_present = 1; |
| swap_in_place(out->sec.ltk, 16); |
| valcnt++; |
| } else if (rc != ENOENT) { |
| console_printf("invalid 'ltk' parameter\n"); |
| return rc; |
| } |
| rc = parse_arg_byte_stream_exact_length("irk", out->sec.irk, 16); |
| if (rc == 0) { |
| out->sec.irk_present = 1; |
| swap_in_place(out->sec.irk, 16); |
| valcnt++; |
| } else if (rc != ENOENT) { |
| console_printf("invalid 'irk' parameter\n"); |
| return rc; |
| } |
| rc = parse_arg_byte_stream_exact_length("csrk", out->sec.csrk, 16); |
| if (rc == 0) { |
| out->sec.csrk_present = 1; |
| swap_in_place(out->sec.csrk, 16); |
| valcnt++; |
| } else if (rc != ENOENT) { |
| console_printf("invalid 'csrk' parameter\n"); |
| return rc; |
| } |
| out->sec.peer_addr = key->sec.peer_addr; |
| break; |
| } |
| |
| if (valcnt) { |
| return 0; |
| } |
| return -1; |
| } |
| |
| /***************************************************************************** |
| * keystore-add * |
| *****************************************************************************/ |
| |
| static int |
| cmd_keystore_add(int argc, char **argv) |
| { |
| union ble_store_key key; |
| union ble_store_value value; |
| int obj_type; |
| int rc; |
| |
| rc = parse_arg_init(argc - 1, argv + 1); |
| if (rc != 0) { |
| return rc; |
| } |
| |
| rc = cmd_keystore_parse_keydata(argc, argv, &key, &obj_type); |
| |
| if (rc) { |
| return rc; |
| } |
| |
| rc = cmd_keystore_parse_valuedata(argc, argv, obj_type, &key, &value); |
| |
| if (rc) { |
| return rc; |
| } |
| |
| switch(obj_type) { |
| case BLE_STORE_OBJ_TYPE_PEER_SEC: |
| rc = ble_store_write_peer_sec(&value.sec); |
| break; |
| case BLE_STORE_OBJ_TYPE_OUR_SEC: |
| rc = ble_store_write_our_sec(&value.sec); |
| break; |
| case BLE_STORE_OBJ_TYPE_CCCD: |
| rc = ble_store_write_cccd(&value.cccd); |
| break; |
| default: |
| rc = ble_store_write(obj_type, &value); |
| } |
| return rc; |
| } |
| |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| static const struct shell_param keystore_add_params[] = { |
| {"type", "entry type, usage: =<msec|ssec|cccd>"}, |
| {"addr_type", "usage: =<public|random>"}, |
| {"addr", "usage: =<XX:XX:XX:XX:XX:XX>"}, |
| {"ediv", "usage: =<UINT16>"}, |
| {"rand", "usage: =<UINT64>"}, |
| {"ltk", "usage: =<XX:XX:...>, len=16 octets"}, |
| {"irk", "usage: =<XX:XX:...>, len=16 octets"}, |
| {"csrk", "usage: =<XX:XX:...>, len=16 octets"}, |
| {NULL, NULL} |
| }; |
| |
| static const struct shell_cmd_help keystore_add_help = { |
| .summary = "add data to keystore", |
| .usage = NULL, |
| .params = keystore_add_params, |
| }; |
| #endif |
| |
| /***************************************************************************** |
| * keystore-del * |
| *****************************************************************************/ |
| |
| static int |
| cmd_keystore_del(int argc, char **argv) |
| { |
| union ble_store_key key; |
| int obj_type; |
| int rc; |
| |
| rc = parse_arg_init(argc - 1, argv + 1); |
| if (rc != 0) { |
| return rc; |
| } |
| |
| rc = cmd_keystore_parse_keydata(argc, argv, &key, &obj_type); |
| |
| if (rc) { |
| return rc; |
| } |
| rc = ble_store_delete(obj_type, &key); |
| return rc; |
| } |
| |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| static const struct shell_param keystore_del_params[] = { |
| {"type", "entry type, usage: =<msec|ssec|cccd>"}, |
| {"addr_type", "usage: =<public|random>"}, |
| {"addr", "usage: =<XX:XX:XX:XX:XX:XX>"}, |
| {"ediv", "usage: =<UINT16>"}, |
| {"rand", "usage: =<UINT64>"}, |
| {NULL, NULL} |
| }; |
| |
| static const struct shell_cmd_help keystore_del_help = { |
| .summary = "remove data from keystore", |
| .usage = NULL, |
| .params = keystore_del_params, |
| }; |
| #endif |
| |
| /***************************************************************************** |
| * keystore-show * |
| *****************************************************************************/ |
| |
| static int |
| cmd_keystore_iterator(int obj_type, |
| union ble_store_value *val, |
| void *cookie) { |
| |
| switch (obj_type) { |
| case BLE_STORE_OBJ_TYPE_PEER_SEC: |
| case BLE_STORE_OBJ_TYPE_OUR_SEC: |
| console_printf("Key: "); |
| if (ble_addr_cmp(&val->sec.peer_addr, BLE_ADDR_ANY) == 0) { |
| console_printf("ediv=%u ", val->sec.ediv); |
| console_printf("ediv=%llu ", val->sec.rand_num); |
| } else { |
| console_printf("addr_type=%u ", val->sec.peer_addr.type); |
| print_addr(val->sec.peer_addr.val); |
| } |
| console_printf("\n"); |
| |
| if (val->sec.ltk_present) { |
| console_printf(" LTK: "); |
| print_bytes(val->sec.ltk, 16); |
| console_printf("\n"); |
| } |
| if (val->sec.irk_present) { |
| console_printf(" IRK: "); |
| print_bytes(val->sec.irk, 16); |
| console_printf("\n"); |
| } |
| if (val->sec.csrk_present) { |
| console_printf(" CSRK: "); |
| print_bytes(val->sec.csrk, 16); |
| console_printf("\n"); |
| } |
| break; |
| case BLE_STORE_OBJ_TYPE_CCCD: |
| console_printf("Key: "); |
| console_printf("addr_type=%u ", val->cccd.peer_addr.type); |
| print_addr(val->cccd.peer_addr.val); |
| console_printf("\n"); |
| |
| console_printf(" char_val_handle: %d\n", val->cccd.chr_val_handle); |
| console_printf(" flags: 0x%02x\n", val->cccd.flags); |
| console_printf(" changed: %d\n", val->cccd.value_changed); |
| break; |
| } |
| return 0; |
| } |
| |
| static int |
| cmd_keystore_show(int argc, char **argv) |
| { |
| int type; |
| int rc; |
| |
| rc = parse_arg_init(argc - 1, argv + 1); |
| if (rc != 0) { |
| return rc; |
| } |
| |
| type = parse_arg_kv("type", cmd_keystore_entry_type, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'type' parameter\n"); |
| return rc; |
| } |
| |
| ble_store_iterate(type, &cmd_keystore_iterator, NULL); |
| return 0; |
| } |
| |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| static const struct shell_param keystore_show_params[] = { |
| {"type", "entry type, usage: =<msec|ssec|cccd>"}, |
| {NULL, NULL} |
| }; |
| |
| static const struct shell_cmd_help keystore_show_help = { |
| .summary = "show data in keystore", |
| .usage = NULL, |
| .params = keystore_show_params, |
| }; |
| #endif |
| |
| #if NIMBLE_BLE_SM |
| /***************************************************************************** |
| * $show-oob-sc * |
| *****************************************************************************/ |
| |
| extern struct ble_sm_sc_oob_data oob_data_local; |
| extern struct ble_sm_sc_oob_data oob_data_remote; |
| |
| static int |
| cmd_show_oob_sc(int argc, char **argv) |
| { |
| console_printf("Local OOB Data: r="); |
| print_bytes(oob_data_local.r, 16); |
| console_printf(" c="); |
| print_bytes(oob_data_local.c, 16); |
| console_printf("\n"); |
| |
| console_printf("Remote OOB Data: r="); |
| print_bytes(oob_data_remote.r, 16); |
| console_printf(" c="); |
| print_bytes(oob_data_remote.c, 16); |
| console_printf("\n"); |
| |
| return 0; |
| } |
| |
| /***************************************************************************** |
| * $auth-passkey * |
| *****************************************************************************/ |
| |
| static int |
| cmd_auth_passkey(int argc, char **argv) |
| { |
| uint16_t conn_handle; |
| struct ble_sm_io pk; |
| char *yesno; |
| int rc; |
| |
| rc = parse_arg_init(argc - 1, argv + 1); |
| if (rc != 0) { |
| return rc; |
| } |
| |
| conn_handle = parse_arg_uint16("conn", &rc); |
| if (rc != 0) { |
| console_printf("invalid 'conn' parameter\n"); |
| return rc; |
| } |
| |
| pk.action = parse_arg_uint16("action", &rc); |
| if (rc != 0) { |
| console_printf("invalid 'action' parameter\n"); |
| return rc; |
| } |
| |
| switch (pk.action) { |
| case BLE_SM_IOACT_INPUT: |
| case BLE_SM_IOACT_DISP: |
| /* passkey is 6 digit number */ |
| pk.passkey = parse_arg_long_bounds("key", 0, 999999, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'key' parameter\n"); |
| return rc; |
| } |
| break; |
| |
| case BLE_SM_IOACT_OOB: |
| rc = parse_arg_byte_stream_exact_length("oob", pk.oob, 16); |
| if (rc != 0) { |
| console_printf("invalid 'oob' parameter\n"); |
| return rc; |
| } |
| break; |
| |
| case BLE_SM_IOACT_NUMCMP: |
| yesno = parse_arg_extract("yesno"); |
| if (yesno == NULL) { |
| console_printf("invalid 'yesno' parameter\n"); |
| return EINVAL; |
| } |
| |
| switch (yesno[0]) { |
| case 'y': |
| case 'Y': |
| pk.numcmp_accept = 1; |
| break; |
| case 'n': |
| case 'N': |
| pk.numcmp_accept = 0; |
| break; |
| |
| default: |
| console_printf("invalid 'yesno' parameter\n"); |
| return EINVAL; |
| } |
| break; |
| |
| case BLE_SM_IOACT_OOB_SC: |
| rc = parse_arg_byte_stream_exact_length("r", oob_data_remote.r, 16); |
| if (rc != 0 && rc != ENOENT) { |
| console_printf("invalid 'r' parameter\n"); |
| return rc; |
| } |
| |
| rc = parse_arg_byte_stream_exact_length("c", oob_data_remote.c, 16); |
| if (rc != 0 && rc != ENOENT) { |
| console_printf("invalid 'c' parameter\n"); |
| return rc; |
| } |
| pk.oob_sc_data.local = &oob_data_local; |
| if (ble_hs_cfg.sm_oob_data_flag) { |
| pk.oob_sc_data.remote = &oob_data_remote; |
| } else { |
| pk.oob_sc_data.remote = NULL; |
| } |
| break; |
| |
| default: |
| console_printf("invalid passkey action action=%d\n", pk.action); |
| return EINVAL; |
| } |
| |
| rc = ble_sm_inject_io(conn_handle, &pk); |
| if (rc != 0) { |
| console_printf("error providing passkey; rc=%d\n", rc); |
| return rc; |
| } |
| |
| return 0; |
| } |
| |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| static const struct shell_param auth_passkey_params[] = { |
| {"conn", "connection handle, usage: =<UINT16>"}, |
| {"action", "auth action type, usage: =<UINT16>"}, |
| {"key", "usage: =[0-999999]"}, |
| {"oob", "usage: =[XX:XX...], len=16 octets"}, |
| {"yesno", "usage: =[string]"}, |
| {NULL, NULL} |
| }; |
| |
| static const struct shell_cmd_help auth_passkey_help = { |
| .summary = "set authorization passkey options", |
| .usage = NULL, |
| .params = auth_passkey_params, |
| }; |
| #endif |
| |
| /***************************************************************************** |
| * $security-pair * |
| *****************************************************************************/ |
| |
| static int |
| cmd_security_pair(int argc, char **argv) |
| { |
| uint16_t conn_handle; |
| int rc; |
| |
| rc = parse_arg_init(argc - 1, argv + 1); |
| if (rc != 0) { |
| return rc; |
| } |
| |
| conn_handle = parse_arg_uint16("conn", &rc); |
| if (rc != 0) { |
| console_printf("invalid 'conn' parameter\n"); |
| return rc; |
| } |
| |
| rc = btshell_sec_pair(conn_handle); |
| if (rc != 0) { |
| console_printf("error initiating pairing; rc=%d\n", rc); |
| return rc; |
| } |
| |
| return 0; |
| } |
| |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| static const struct shell_param security_pair_params[] = { |
| {"conn", "connection handle, usage: =<UINT16>"}, |
| {NULL, NULL} |
| }; |
| |
| static const struct shell_cmd_help security_pair_help = { |
| .summary = "start pairing procedure for connection", |
| .usage = NULL, |
| .params = security_pair_params, |
| }; |
| #endif |
| |
| /***************************************************************************** |
| * $security-unpair * |
| *****************************************************************************/ |
| |
| static int |
| cmd_security_unpair(int argc, char **argv) |
| { |
| ble_addr_t peer; |
| int rc; |
| int oldest; |
| |
| rc = parse_arg_init(argc - 1, argv + 1); |
| if (rc != 0) { |
| return rc; |
| } |
| |
| rc = parse_arg_bool_dflt("oldest", 0, &oldest); |
| if (rc != 0) { |
| console_printf("invalid 'oldest' parameter\n"); |
| return rc; |
| } |
| |
| if (oldest) { |
| rc = ble_gap_unpair_oldest_peer(); |
| console_printf("Unpair oldest status: 0x%02x\n", rc); |
| return 0; |
| } |
| |
| rc = parse_dev_addr("peer_", cmd_peer_addr_types, &peer); |
| if (rc != 0) { |
| console_printf("invalid peer address\n"); |
| return rc; |
| } |
| |
| rc = ble_gap_unpair(&peer); |
| if (rc != 0) { |
| console_printf("error unpairing; rc=%d\n", rc); |
| return rc; |
| } |
| |
| return 0; |
| } |
| |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| static const struct shell_param security_unpair_params[] = { |
| {"oldest", "usage: =[true|false], default: false"}, |
| {"peer_addr_type", "usage: =[public|random|public_id|random_id], default: public"}, |
| {"peer_addr", "usage: =[XX:XX:XX:XX:XX:XX]"}, |
| {NULL, NULL} |
| }; |
| |
| static const struct shell_cmd_help security_unpair_help = { |
| .summary = "unpair a peer device", |
| .usage = NULL, |
| .params = security_unpair_params, |
| }; |
| #endif |
| |
| /***************************************************************************** |
| * $security-start * |
| *****************************************************************************/ |
| |
| static int |
| cmd_security_start(int argc, char **argv) |
| { |
| uint16_t conn_handle; |
| int rc; |
| |
| rc = parse_arg_init(argc - 1, argv + 1); |
| if (rc != 0) { |
| return rc; |
| } |
| |
| conn_handle = parse_arg_uint16("conn", &rc); |
| if (rc != 0) { |
| console_printf("invalid 'conn' parameter\n"); |
| return rc; |
| } |
| |
| rc = btshell_sec_start(conn_handle); |
| if (rc != 0) { |
| console_printf("error starting security; rc=%d\n", rc); |
| return rc; |
| } |
| |
| return 0; |
| } |
| |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| static const struct shell_param security_start_params[] = { |
| {"conn", "connection handle, usage: =<UINT16>"}, |
| {NULL, NULL} |
| }; |
| |
| static const struct shell_cmd_help security_start_help = { |
| .summary = "start security procedure for connection", |
| .usage = NULL, |
| .params = security_start_params, |
| }; |
| #endif |
| |
| /***************************************************************************** |
| * $security-encryption * |
| *****************************************************************************/ |
| |
| static int |
| cmd_security_encryption(int argc, char **argv) |
| { |
| uint16_t conn_handle; |
| uint16_t ediv; |
| uint64_t rand_val; |
| uint8_t ltk[16]; |
| uint8_t key_size; |
| int rc; |
| int auth; |
| |
| rc = parse_arg_init(argc - 1, argv + 1); |
| if (rc != 0) { |
| return rc; |
| } |
| |
| conn_handle = parse_arg_uint16("conn", &rc); |
| if (rc != 0) { |
| console_printf("invalid 'conn' parameter\n"); |
| return rc; |
| } |
| |
| ediv = parse_arg_uint16("ediv", &rc); |
| if (rc == ENOENT) { |
| rc = btshell_sec_restart(conn_handle, 0, NULL, 0, 0, 0); |
| } else { |
| rand_val = parse_arg_uint64("rand", &rc); |
| if (rc != 0) { |
| console_printf("invalid 'rand' parameter\n"); |
| return rc; |
| } |
| |
| auth = parse_arg_bool("auth", &rc); |
| if (rc != 0) { |
| console_printf("invalid 'auth' parameter\n"); |
| return rc; |
| } |
| |
| key_size = parse_arg_uint8("key_size", &rc); |
| if (rc != 0) { |
| console_printf("invalid 'key_size' parameter\n"); |
| return rc; |
| } |
| |
| rc = parse_arg_byte_stream_exact_length("ltk", ltk, 16); |
| if (rc != 0) { |
| console_printf("invalid 'ltk' parameter\n"); |
| return rc; |
| } |
| |
| rc = btshell_sec_restart(conn_handle, key_size, |
| ltk, ediv, rand_val, auth); |
| } |
| |
| if (rc != 0) { |
| console_printf("error initiating encryption; rc=%d\n", rc); |
| return rc; |
| } |
| |
| return 0; |
| } |
| |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| static const struct shell_param security_encryption_params[] = { |
| {"conn", "connection handle, usage: =<UINT16>"}, |
| {"ediv", "usage: =[UINT16]"}, |
| {"rand", "usage: =[UINT64]"}, |
| {"auth", "usage: =[0-1]"}, |
| {"ltk", "usage: =[XX:XX...], len=16 octets"}, |
| {NULL, NULL} |
| }; |
| |
| static const struct shell_cmd_help security_encryption_help = { |
| .summary = "start encryption procedure for connection", |
| .usage = NULL, |
| .params = security_encryption_params, |
| }; |
| #endif |
| |
| /***************************************************************************** |
| * $security-set-data * |
| *****************************************************************************/ |
| |
| static int |
| cmd_security_set_data(int argc, char **argv) |
| { |
| uint8_t tmp; |
| int good; |
| int rc; |
| |
| good = 0; |
| |
| rc = parse_arg_init(argc - 1, argv + 1); |
| if (rc != 0) { |
| return rc; |
| } |
| |
| tmp = parse_arg_bool("oob_flag", &rc); |
| if (rc == 0) { |
| ble_hs_cfg.sm_oob_data_flag = tmp; |
| good++; |
| } else if (rc != ENOENT) { |
| console_printf("invalid 'oob_flag' parameter\n"); |
| return rc; |
| } |
| |
| tmp = parse_arg_bool("mitm_flag", &rc); |
| if (rc == 0) { |
| good++; |
| ble_hs_cfg.sm_mitm = tmp; |
| } else if (rc != ENOENT) { |
| console_printf("invalid 'mitm_flag' parameter\n"); |
| return rc; |
| } |
| |
| tmp = parse_arg_uint8("io_capabilities", &rc); |
| if (rc == 0) { |
| good++; |
| ble_hs_cfg.sm_io_cap = tmp; |
| } else if (rc != ENOENT) { |
| console_printf("invalid 'io_capabilities' parameter\n"); |
| return rc; |
| } |
| |
| tmp = parse_arg_uint8("our_key_dist", &rc); |
| if (rc == 0) { |
| good++; |
| ble_hs_cfg.sm_our_key_dist = tmp; |
| } else if (rc != ENOENT) { |
| console_printf("invalid 'our_key_dist' parameter\n"); |
| return rc; |
| } |
| |
| tmp = parse_arg_uint8("their_key_dist", &rc); |
| if (rc == 0) { |
| good++; |
| ble_hs_cfg.sm_their_key_dist = tmp; |
| } else if (rc != ENOENT) { |
| console_printf("invalid 'their_key_dist' parameter\n"); |
| return rc; |
| } |
| |
| tmp = parse_arg_bool("bonding", &rc); |
| if (rc == 0) { |
| good++; |
| ble_hs_cfg.sm_bonding = tmp; |
| } else if (rc != ENOENT) { |
| console_printf("invalid 'bonding' parameter\n"); |
| return rc; |
| } |
| |
| tmp = parse_arg_bool("sc", &rc); |
| if (rc == 0) { |
| good++; |
| ble_hs_cfg.sm_sc = tmp; |
| } else if (rc != ENOENT) { |
| console_printf("invalid 'sc' parameter\n"); |
| return rc; |
| } |
| |
| if (!good) { |
| console_printf("Error: no valid settings specified\n"); |
| return -1; |
| } |
| |
| return 0; |
| } |
| |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| static const struct shell_param security_set_data_params[] = { |
| {"oob_flag", "usage: =[0-1]"}, |
| {"mitm_flag", "usage: =[0-1]"}, |
| {"io_capabilities", "usage: =[UINT8]"}, |
| {"our_key_dist", "usage: =[UINT8]"}, |
| {"their_key_dist", "usage: =[UINT8]"}, |
| {"bonding", "usage: =[0-1]"}, |
| {"sc", "usage: =[0-1]"}, |
| {NULL, NULL} |
| }; |
| |
| static const struct shell_cmd_help security_set_data_help = { |
| .summary = "set security data", |
| .usage = NULL, |
| .params = security_set_data_params, |
| }; |
| #endif |
| #endif |
| |
| /***************************************************************************** |
| * $test-tx * |
| * * |
| * Command to transmit 'num' packets of size 'len' at rate 'r' to |
| * handle 'h' Note that length must be <= 251. The rate is in msecs. |
| * |
| *****************************************************************************/ |
| |
| static int |
| cmd_test_tx(int argc, char **argv) |
| { |
| int rc; |
| uint16_t conn; |
| uint16_t len; |
| uint16_t rate; |
| uint16_t num; |
| uint8_t stop; |
| |
| rc = parse_arg_init(argc - 1, argv + 1); |
| if (rc != 0) { |
| return rc; |
| } |
| |
| stop = parse_arg_uint8_dflt("stop", 0, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'stop' parameter\n"); |
| return rc; |
| } |
| |
| if (stop) { |
| btshell_tx_stop(); |
| return 0; |
| } |
| |
| conn = parse_arg_uint16("conn", &rc); |
| if (rc != 0) { |
| console_printf("invalid 'conn' parameter\n"); |
| return rc; |
| } |
| |
| len = parse_arg_uint16("length", &rc); |
| if (rc != 0) { |
| console_printf("invalid 'length' parameter\n"); |
| return rc; |
| } |
| if ((len > 251) || (len < 4)) { |
| console_printf("error: len must be between 4 and 251, inclusive"); |
| } |
| |
| rate = parse_arg_uint16_dflt("rate", 1, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'rate' parameter\n"); |
| return rc; |
| } |
| |
| num = parse_arg_uint16_dflt("num", 1, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'num' parameter\n"); |
| return rc; |
| } |
| |
| rc = btshell_tx_start(conn, len, rate, num); |
| return rc; |
| } |
| |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| static const struct shell_param test_tx_params[] = { |
| {"conn", "handle to tx to, usage: =<UINT16>"}, |
| {"length", "size of packet, usage: =<UINT16>"}, |
| {"rate", "rate of tx, usage: =<UINT16>, default=1"}, |
| {"num", "number of packets, usage: =<UINT16>, default=1"}, |
| {"stop", "stop sending, usage: 1 to stop, default 0"}, |
| {NULL, NULL} |
| }; |
| |
| static const struct shell_cmd_help test_tx_help = { |
| .summary = "test packet transmission", |
| .usage = NULL, |
| .params = test_tx_params, |
| }; |
| #endif |
| |
| /***************************************************************************** |
| * $phy-set * |
| *****************************************************************************/ |
| |
| static int |
| cmd_phy_set(int argc, char **argv) |
| { |
| uint16_t conn; |
| uint8_t tx_phys_mask; |
| uint8_t rx_phys_mask; |
| uint16_t phy_opts; |
| int rc; |
| |
| rc = parse_arg_init(argc - 1, argv + 1); |
| if (rc != 0) { |
| return rc; |
| } |
| |
| conn = parse_arg_uint16("conn", &rc); |
| if (rc != 0) { |
| console_printf("invalid 'conn' parameter\n"); |
| return rc; |
| } |
| |
| tx_phys_mask = parse_arg_uint8("tx_phys_mask", &rc); |
| if (rc != 0) { |
| console_printf("invalid 'tx_phys_mask' parameter\n"); |
| return rc; |
| } |
| |
| rx_phys_mask = parse_arg_uint8("rx_phys_mask", &rc); |
| if (rc != 0) { |
| console_printf("invalid 'rx_phys_mask' parameter\n"); |
| return rc; |
| } |
| |
| phy_opts = parse_arg_uint16("phy_opts", &rc); |
| if (rc != 0) { |
| console_printf("invalid 'phy_opts' parameter\n"); |
| return rc; |
| } |
| |
| return ble_gap_set_prefered_le_phy(conn, tx_phys_mask, rx_phys_mask, |
| phy_opts); |
| } |
| |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| static const struct shell_param phy_set_params[] = { |
| {"conn", "connection handle, usage: =<UINT16>"}, |
| {"tx_phys_mask", "usage: =<UINT8>"}, |
| {"rx_phys_mask", "usage: =<UINT8>"}, |
| {"phy_opts", "usage: =<UINT16>"}, |
| {NULL, NULL} |
| }; |
| |
| static const struct shell_cmd_help phy_set_help = { |
| .summary = "set preferred PHYs", |
| .usage = NULL, |
| .params = phy_set_params, |
| }; |
| #endif |
| |
| /***************************************************************************** |
| * $phy-set-default * |
| *****************************************************************************/ |
| |
| static int |
| cmd_phy_set_default(int argc, char **argv) |
| { |
| uint8_t tx_phys_mask; |
| uint8_t rx_phys_mask; |
| int rc; |
| |
| rc = parse_arg_init(argc - 1, argv + 1); |
| if (rc != 0) { |
| return rc; |
| } |
| |
| tx_phys_mask = parse_arg_uint8("tx_phys_mask", &rc); |
| if (rc != 0) { |
| console_printf("invalid 'tx_phys_mask' parameter\n"); |
| return rc; |
| } |
| |
| rx_phys_mask = parse_arg_uint8("rx_phys_mask", &rc); |
| if (rc != 0) { |
| console_printf("invalid 'rx_phys_mask' parameter\n"); |
| return rc; |
| } |
| |
| return ble_gap_set_prefered_default_le_phy(tx_phys_mask, rx_phys_mask); |
| } |
| |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| static const struct shell_param phy_set_default_params[] = { |
| {"tx_phys_mask", "usage: =<UINT8>"}, |
| {"rx_phys_mask", "usage: =<UINT8>"}, |
| {NULL, NULL} |
| }; |
| |
| static const struct shell_cmd_help phy_set_default_help = { |
| .summary = "set preferred default PHYs", |
| .usage = NULL, |
| .params = phy_set_default_params, |
| }; |
| #endif |
| |
| /***************************************************************************** |
| * $phy-read * |
| *****************************************************************************/ |
| |
| static int |
| cmd_phy_read(int argc, char **argv) |
| { |
| uint16_t conn = 0; |
| uint8_t tx_phy; |
| uint8_t rx_phy; |
| int rc; |
| |
| rc = parse_arg_init(argc - 1, argv + 1); |
| if (rc != 0) { |
| return rc; |
| } |
| |
| conn = parse_arg_uint16("conn", &rc); |
| if (rc != 0) { |
| console_printf("invalid 'conn' parameter\n"); |
| return rc; |
| } |
| |
| rc = ble_gap_read_le_phy(conn, &tx_phy, &rx_phy); |
| if (rc != 0) { |
| console_printf("Could not read PHY error: %d\n", rc); |
| return rc; |
| } |
| |
| console_printf("TX_PHY: %d\n", tx_phy); |
| console_printf("RX_PHY: %d\n", tx_phy); |
| |
| return 0; |
| } |
| |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| static const struct shell_param phy_read_params[] = { |
| {"conn", "connection handle, usage: =<UINT16>"}, |
| {NULL, NULL} |
| }; |
| |
| static const struct shell_cmd_help phy_read_help = { |
| .summary = "read PHYs", |
| .usage = NULL, |
| .params = phy_read_params, |
| }; |
| #endif |
| |
| /***************************************************************************** |
| * $host-enable * |
| *****************************************************************************/ |
| |
| static int |
| cmd_host_enable(int argc, char **argv) |
| { |
| int rc; |
| |
| rc = gatt_svr_init(); |
| assert(rc == 0); |
| |
| ble_hs_sched_start(); |
| |
| return 0; |
| } |
| |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| static const struct shell_cmd_help host_enable_help = { |
| .summary = "start the NimBLE host", |
| .usage = NULL, |
| .params = NULL, |
| }; |
| #endif |
| |
| /***************************************************************************** |
| * $host-disable * |
| *****************************************************************************/ |
| |
| static void |
| on_stop(int status, void *arg) |
| { |
| if (status == 0) { |
| console_printf("host stopped\n"); |
| } else { |
| console_printf("host failed to stop; rc=%d\n", status); |
| } |
| } |
| |
| static int |
| cmd_host_disable(int argc, char **argv) |
| { |
| static struct ble_hs_stop_listener listener; |
| int rc; |
| |
| rc = ble_hs_stop(&listener, on_stop, NULL); |
| if (rc) { |
| return rc; |
| } |
| |
| ble_gatts_reset(); |
| |
| return 0; |
| } |
| |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| static const struct shell_cmd_help host_disable_help = { |
| .summary = "stop the NimBLE host", |
| .usage = NULL, |
| .params = NULL, |
| }; |
| |
| /***************************************************************************** |
| * $gatt-discover * |
| *****************************************************************************/ |
| |
| static const struct shell_param gatt_discover_characteristic_params[] = { |
| {"conn", "connection handle, usage: =<UINT16>"}, |
| {"uuid", "discover by uuid, usage: =[UUID]"}, |
| {"start", "start handle, usage: =<UINT16>"}, |
| {"end", "end handle, usage: =<UINT16>"}, |
| {NULL, NULL} |
| }; |
| |
| static const struct shell_cmd_help gatt_discover_characteristic_help = { |
| .summary = "perform characteristic discovery procedure", |
| .usage = NULL, |
| .params = gatt_discover_characteristic_params, |
| }; |
| |
| static const struct shell_param gatt_discover_descriptor_params[] = { |
| {"conn", "connection handle, usage: =<UINT16>"}, |
| {"start", "start handle, usage: =<UINT16>"}, |
| {"end", "end handle, usage: =<UINT16>"}, |
| {NULL, NULL} |
| }; |
| |
| static const struct shell_cmd_help gatt_discover_descriptor_help = { |
| .summary = "perform descriptor discovery procedure", |
| .usage = NULL, |
| .params = gatt_discover_descriptor_params, |
| }; |
| |
| static const struct shell_param gatt_discover_service_params[] = { |
| {"conn", "connection handle, usage: =<UINT16>"}, |
| {"uuid", "discover by uuid, usage: =[UUID]"}, |
| {NULL, NULL} |
| }; |
| |
| static const struct shell_cmd_help gatt_discover_service_help = { |
| .summary = "perform service discovery procedure", |
| .usage = NULL, |
| .params = gatt_discover_service_params, |
| }; |
| |
| static const struct shell_param gatt_discover_full_params[] = { |
| {"conn", "connection handle, usage: =<UINT16>"}, |
| {NULL, NULL} |
| }; |
| |
| static const struct shell_cmd_help gatt_discover_full_help = { |
| .summary = "perform full discovery procedure", |
| .usage = NULL, |
| .params = gatt_discover_full_params, |
| }; |
| |
| /***************************************************************************** |
| * $gatt-exchange-mtu * |
| *****************************************************************************/ |
| |
| static const struct shell_param gatt_exchange_mtu_params[] = { |
| {"conn", "connection handle, usage: =<UINT16>"}, |
| {NULL, NULL} |
| }; |
| |
| static const struct shell_cmd_help gatt_exchange_mtu_help = { |
| .summary = "perform mtu exchange procedure", |
| .usage = NULL, |
| .params = gatt_exchange_mtu_params, |
| }; |
| |
| /***************************************************************************** |
| * $gatt-find-included-services * |
| *****************************************************************************/ |
| |
| static const struct shell_param gatt_find_included_services_params[] = { |
| {"conn", "connection handle, usage: =<UINT16>"}, |
| {"start", "start handle, usage: =<UINT16>"}, |
| {"end", "end handle, usage: =<UINT16>"}, |
| {NULL, NULL} |
| }; |
| |
| static const struct shell_cmd_help gatt_find_included_services_help = { |
| .summary = "perform find included services procedure", |
| .usage = NULL, |
| .params = gatt_find_included_services_params, |
| }; |
| |
| /***************************************************************************** |
| * $gatt-notify * |
| *****************************************************************************/ |
| |
| static const struct shell_param gatt_notify_params[] = { |
| {"attr", "attribute handle, usage: =<UINT16>"}, |
| {NULL, NULL} |
| }; |
| |
| static const struct shell_cmd_help gatt_notify_help = { |
| .summary = "notify about attribute value changed", |
| .usage = NULL, |
| .params = gatt_notify_params, |
| }; |
| |
| /***************************************************************************** |
| * $gatt-read * |
| *****************************************************************************/ |
| |
| static const struct shell_param gatt_read_params[] = { |
| {"conn", "connection handle, usage: =<UINT16>"}, |
| {"long", "is read long, usage: =[0-1], default=0"}, |
| {"attr", "attribute handle, usage: =<UINT16>"}, |
| {"offset", "offset value, usage: =<UINT16>"}, |
| {"uuid", "read by uuid, usage: =[UUID]"}, |
| {"start", "start handle, usage: =<UINT16>"}, |
| {"end", "end handle, usage: =<UINT16>"}, |
| {"variable", "used in case of multi read, usage: =[0-1], default=0"}, |
| {NULL, NULL} |
| }; |
| |
| static const struct shell_cmd_help gatt_read_help = { |
| .summary = "perform gatt read procedure", |
| .usage = NULL, |
| .params = gatt_read_params, |
| }; |
| |
| /***************************************************************************** |
| * $gatt-service-changed * |
| *****************************************************************************/ |
| |
| static const struct shell_param gatt_service_changed_params[] = { |
| {"start", "start handle, usage: =<UINT16>"}, |
| {"end", "end handle, usage: =<UINT16>"}, |
| {NULL, NULL} |
| }; |
| |
| static const struct shell_cmd_help gatt_service_changed_help = { |
| .summary = "send service changed indication", |
| .usage = NULL, |
| .params = gatt_service_changed_params, |
| }; |
| |
| /***************************************************************************** |
| * $gatt-service-visibility * |
| *****************************************************************************/ |
| |
| static const struct shell_param gatt_service_visibility_params[] = { |
| {"handle", "usage: =<UINT16>"}, |
| {"visibility", "usage: =<0-1>"}, |
| {NULL, NULL} |
| }; |
| |
| static const struct shell_cmd_help gatt_service_visibility_help = { |
| .summary = "change service visibility", |
| .usage = NULL, |
| .params = gatt_service_visibility_params, |
| }; |
| |
| /***************************************************************************** |
| * $gatt-show * |
| *****************************************************************************/ |
| |
| static const struct shell_param gatt_show_params[] = { |
| {NULL, NULL} |
| }; |
| |
| static const struct shell_cmd_help gatt_show_help = { |
| .summary = "show discovered gatt database", |
| .usage = NULL, |
| .params = gatt_show_params, |
| }; |
| |
| static const struct shell_cmd_help gatt_show_local_help = { |
| .summary = "show local gatt database", |
| .usage = NULL, |
| .params = gatt_show_params, |
| }; |
| |
| static const struct shell_cmd_help gatt_show_addr_help = { |
| .summary = "show device address", |
| .usage = NULL, |
| .params = gatt_show_params, |
| }; |
| |
| static const struct shell_cmd_help gatt_show_conn_help = { |
| .summary = "show connections information", |
| .usage = NULL, |
| .params = gatt_show_params, |
| }; |
| |
| /***************************************************************************** |
| * $gatt-write * |
| *****************************************************************************/ |
| |
| static const struct shell_param gatt_write_params[] = { |
| {"conn", "connection handle, usage: =<UINT16>"}, |
| {"no_rsp", "write without response, usage: =[0-1], default=0"}, |
| {"long", "is write long, usage: =[0-1], default=0"}, |
| {"attr", "attribute handle, usage: =<UINT16>"}, |
| {"offset", "attribute handle, usage: =<UINT16>"}, |
| {"value", "usage: =<octets>"}, |
| {NULL, NULL} |
| }; |
| |
| static const struct shell_cmd_help gatt_write_help = { |
| .summary = "perform gatt write procedure", |
| .usage = NULL, |
| .params = gatt_write_params, |
| }; |
| |
| /***************************************************************************** |
| * $gatt-enqueue-notify * |
| *****************************************************************************/ |
| |
| static const struct shell_param gatt_enqueue_notif_params[] = { |
| {"handle", "characteristic handle, usage: =<UINT16>"}, |
| {"value", "usage: =<octets>"}, |
| {NULL, NULL} |
| }; |
| |
| static const struct shell_cmd_help gatt_enqueue_notif_help = { |
| .summary = "enqueue notification to be sent", |
| .usage = NULL, |
| .params = gatt_enqueue_notif_params, |
| }; |
| |
| /***************************************************************************** |
| * $gatt-send-pending-notify * |
| *****************************************************************************/ |
| |
| static const struct shell_param gatt_send_pending_notif_params[] = { |
| {"conn", "connection handle, usage: =<UINT16>"}, |
| {NULL, NULL} |
| }; |
| |
| static const struct shell_cmd_help gatt_send_pending_notif_help = { |
| .summary = "send pending notifications", |
| .usage = NULL, |
| .params = gatt_send_pending_notif_params, |
| }; |
| |
| static const struct shell_cmd_help gatt_clear_pending_notif_help = { |
| .summary = "clear pending notifications", |
| .usage = NULL, |
| .params = NULL, |
| }; |
| #endif |
| |
| #if MYNEWT_VAL(BLE_L2CAP_COC_MAX_NUM) |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| /***************************************************************************** |
| * $l2cap-update * |
| *****************************************************************************/ |
| |
| static const struct shell_param l2cap_update_params[] = { |
| {"conn", "connection handle, usage: =<UINT16>"}, |
| {"interval_min", "usage: =[0-UINT16_MAX], default: 30"}, |
| {"interval_max", "usage: =[0-UINT16_MAX], default: 50"}, |
| {"latency", "usage: =[UINT16], default: 0"}, |
| {"timeout", "usage: =[UINT16], default: 0x0100"}, |
| {NULL, NULL} |
| }; |
| |
| static const struct shell_cmd_help l2cap_update_help = { |
| .summary = "update l2cap parameters for connection", |
| .usage = NULL, |
| .params = l2cap_update_params, |
| }; |
| |
| /***************************************************************************** |
| * $l2cap-create-server * |
| *****************************************************************************/ |
| |
| static const struct shell_param l2cap_create_server_params[] = { |
| {"psm", "usage: =<UINT16>"}, |
| {"mtu", "usage: =<UINT16> not more than BTSHELL_COC_MTU, default BTSHELL_COC_MTU"}, |
| {"error", "usage: used for PTS testing:"}, |
| {"", "0 - always accept"}, |
| {"", "1 - reject with insufficient authentication"}, |
| {"", "2 - reject with insufficient authorization"}, |
| {"", "3 - reject with insufficient key size"}, |
| {NULL, NULL} |
| }; |
| |
| static const struct shell_cmd_help l2cap_create_server_help = { |
| .summary = "create l2cap server", |
| .usage = NULL, |
| .params = l2cap_create_server_params, |
| }; |
| |
| /***************************************************************************** |
| * $l2cap-connect * |
| *****************************************************************************/ |
| |
| static const struct shell_param l2cap_connect_params[] = { |
| {"conn", "connection handle, usage: =<UINT16>"}, |
| {"psm", "usage: =<UINT16>"}, |
| {"num", "usage: number of connection created in a row: [1-5]"}, |
| {"mtu", "usage: =<UINT16> not more than BTSHELL_COC_MTU, default BTSHELL_COC_MTU"}, |
| {NULL, NULL} |
| }; |
| |
| static const struct shell_cmd_help l2cap_connect_help = { |
| .summary = "perform l2cap connect procedure", |
| .usage = NULL, |
| .params = l2cap_connect_params, |
| }; |
| |
| /***************************************************************************** |
| * $l2cap-disconnect * |
| *****************************************************************************/ |
| |
| static const struct shell_param l2cap_disconnect_params[] = { |
| {"conn", "connection handle, usage: =<UINT16>"}, |
| {"idx", "usage: =<UINT16>"}, |
| {NULL, NULL} |
| }; |
| |
| static const struct shell_cmd_help l2cap_disconnect_help = { |
| .summary = "perform l2cap disconnect procedure", |
| .usage = "use gatt-show-coc to get the parameters", |
| .params = l2cap_disconnect_params, |
| }; |
| |
| /***************************************************************************** |
| * $l2cap-reconfig * |
| *****************************************************************************/ |
| |
| static const struct shell_param l2cap_reconfig_params[] = { |
| {"conn", "connection handle, usage: =<UINT16>"}, |
| {"mtu", "new mtu, usage: =<UINT16>, default: 0 (no change)"}, |
| {"idxs", "list of channel indexes, usage: idxs=1,3"}, |
| {NULL, NULL} |
| }; |
| |
| static const struct shell_cmd_help l2cap_reconfig_help = { |
| .summary = "perform l2cap reconfigure procedure", |
| .usage = "use gatt-show-coc to get the parameters", |
| .params = l2cap_reconfig_params, |
| }; |
| |
| /***************************************************************************** |
| * $l2cap-send * |
| *****************************************************************************/ |
| |
| static const struct shell_param l2cap_send_params[] = { |
| {"conn", "connection handle, usage: =<UINT16>"}, |
| {"idx", "usage: =<UINT16>"}, |
| {"bytes", "number of bytes to send, usage: =<UINT16>"}, |
| {NULL, NULL} |
| }; |
| |
| static const struct shell_cmd_help l2cap_send_help = { |
| .summary = "perform l2cap send procedure", |
| .usage = "use l2cap-show-coc to get the parameters", |
| .params = l2cap_send_params, |
| }; |
| |
| /***************************************************************************** |
| * $l2cap-show-coc * |
| *****************************************************************************/ |
| |
| static const struct shell_param l2cap_show_coc_params[] = { |
| {NULL, NULL} |
| }; |
| |
| static const struct shell_cmd_help l2cap_show_coc_help = { |
| .summary = "show coc information", |
| .usage = NULL, |
| .params = l2cap_show_coc_params, |
| }; |
| |
| #endif |
| #endif |
| |
| #if MYNEWT_VAL(BLE_PERIODIC_ADV) |
| static int |
| cmd_periodic_configure(int argc, char **argv) |
| { |
| struct ble_gap_periodic_adv_params params = {0}; |
| uint8_t instance; |
| int rc; |
| |
| rc = parse_arg_init(argc - 1, argv + 1); |
| if (rc != 0) { |
| return rc; |
| } |
| |
| instance = parse_arg_uint8_dflt("instance", 0, &rc); |
| if (rc != 0 || instance >= BLE_ADV_INSTANCES) { |
| console_printf("invalid instance\n"); |
| return rc; |
| } |
| |
| memset(¶ms, 0, sizeof(params)); |
| |
| params.include_tx_power = parse_arg_bool_dflt("include_tx_power", 0, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'include_tx_power' parameter\n"); |
| return rc; |
| } |
| |
| params.itvl_min = parse_arg_time_dflt("interval_min", 1250, 0, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'interval_min' parameter\n"); |
| return rc; |
| } |
| |
| params.itvl_max = parse_arg_time_dflt("interval_max", 1250, params.itvl_min, |
| &rc); |
| if (rc != 0) { |
| console_printf("invalid 'interval_max' parameter\n"); |
| return rc; |
| } |
| |
| rc = ble_gap_periodic_adv_configure(instance, ¶ms); |
| if (rc) { |
| console_printf("failed to configure periodic advertising\n"); |
| return rc; |
| } |
| |
| console_printf("Instance %u configured for periodic advertising\n", |
| instance); |
| |
| return 0; |
| } |
| |
| static int |
| cmd_periodic_set_adv_data(int argc, char **argv) |
| { |
| return cmd_set_adv_data_or_scan_rsp(argc, argv, false, true); |
| } |
| |
| static int |
| cmd_periodic_start(int argc, char **argv) |
| { |
| uint8_t instance; |
| int rc; |
| |
| rc = parse_arg_init(argc - 1, argv + 1); |
| if (rc != 0) { |
| return rc; |
| } |
| |
| instance = parse_arg_uint8_dflt("instance", 0, &rc); |
| if (rc != 0 || instance >= BLE_ADV_INSTANCES) { |
| console_printf("invalid instance\n"); |
| return rc; |
| } |
| |
| rc = ble_gap_periodic_adv_start(instance, NULL); |
| if (rc) { |
| console_printf("failed to start periodic advertising\n"); |
| return rc; |
| } |
| |
| return 0; |
| } |
| |
| static int |
| cmd_periodic_stop(int argc, char **argv) |
| { |
| uint8_t instance; |
| int rc; |
| |
| rc = parse_arg_init(argc - 1, argv + 1); |
| if (rc != 0) { |
| return rc; |
| } |
| |
| instance = parse_arg_uint8_dflt("instance", 0, &rc); |
| if (rc != 0 || instance >= BLE_ADV_INSTANCES) { |
| console_printf("invalid instance\n"); |
| return rc; |
| } |
| |
| rc = ble_gap_periodic_adv_stop(instance); |
| if (rc) { |
| console_printf("failed to stop periodic advertising\n"); |
| return rc; |
| } |
| |
| return 0; |
| } |
| |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| static const struct shell_param periodic_configure_params[] = { |
| {"instance", "default: 0"}, |
| {"interval_min", "usage: =[0-UINT32_MAX], default: 0"}, |
| {"interval_max", "usage: =[0-UINT32_MAX], default: interval_min"}, |
| {"tx_power", "include TX power, usage: =[0-1], default: 0"}, |
| {NULL, NULL} |
| }; |
| |
| static const struct shell_cmd_help periodic_configure_help = { |
| .summary = "configure periodic advertising for instance", |
| .usage = NULL, |
| .params = periodic_configure_params, |
| }; |
| |
| static const struct shell_param periodic_start_params[] = { |
| {"instance", "default: 0"}, |
| {NULL, NULL} |
| }; |
| |
| static const struct shell_cmd_help periodic_start_help = { |
| .summary = "start periodic advertising for instance", |
| .usage = NULL, |
| .params = periodic_start_params, |
| }; |
| |
| static const struct shell_param periodic_stop_params[] = { |
| {"instance", "default: 0"}, |
| {NULL, NULL} |
| }; |
| |
| static const struct shell_cmd_help periodic_stop_help = { |
| .summary = "stop periodic advertising for instance", |
| .usage = NULL, |
| .params = periodic_stop_params, |
| }; |
| #endif |
| |
| static int |
| cmd_sync_create(int argc, char **argv) |
| { |
| struct ble_gap_periodic_sync_params params = { 0 }; |
| ble_addr_t addr; |
| ble_addr_t *addr_param = &addr; |
| uint8_t sid; |
| int rc; |
| |
| rc = parse_arg_init(argc - 1, argv + 1); |
| if (rc != 0) { |
| return rc; |
| } |
| |
| if (argc > 1 && strcmp(argv[1], "cancel") == 0) { |
| rc = ble_gap_periodic_adv_sync_create_cancel(); |
| if (rc != 0) { |
| console_printf("Sync create cancel fail: %d\n", rc); |
| return rc; |
| } |
| |
| return 0; |
| } |
| |
| rc = parse_dev_addr("peer_", cmd_addr_type, &addr); |
| if (rc == ENOENT) { |
| /* With no "peer_addr" specified we'll use periodic list */ |
| addr_param = NULL; |
| } else if (rc != 0) { |
| console_printf("invalid 'addr' parameter\n"); |
| return rc; |
| } |
| |
| sid = parse_arg_uint8_dflt("sid", 0, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'sid' parameter\n"); |
| return rc; |
| } |
| |
| params.skip = parse_arg_uint16_dflt("skip", 0, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'skip' parameter\n"); |
| return rc; |
| } |
| |
| params.sync_timeout = parse_arg_time_dflt("sync_timeout", 10000, 2000, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'sync_timeout' parameter\n"); |
| return rc; |
| } |
| |
| params.reports_disabled = parse_arg_bool_dflt("reports_disabled", false, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'reports_disabled' parameter\n"); |
| return rc; |
| } |
| |
| rc = ble_gap_periodic_adv_sync_create(addr_param, sid, ¶ms, |
| btshell_gap_event, NULL); |
| if (rc) { |
| console_printf("Failed to create sync (%d)\n", rc); |
| } |
| |
| return rc; |
| } |
| |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| static const struct shell_param sync_create_params[] = { |
| {"cancel", "cancel periodic sync establishment procedure"}, |
| {"peer_addr_type", "usage: =[public|random], default: public"}, |
| {"peer_addr", "usage: =[XX:XX:XX:XX:XX:XX]"}, |
| {"sid", "usage: =[UINT8], default: 0"}, |
| {"skip", "usage: =[0-0x01F3], default: 0x0000"}, |
| {"sync_timeout", "usage: =[0x000A-0x4000], default: 0x07D0"}, |
| {"reports_disabled", "disable reports, usage: =[0-1], default: 0"}, |
| {NULL, NULL} |
| }; |
| |
| static const struct shell_cmd_help sync_create_help = { |
| .summary = "start/stop periodic sync procedure with specific parameters", |
| .usage = NULL, |
| .params = sync_create_params, |
| }; |
| #endif |
| |
| #if MYNEWT_VAL(BLE_PERIODIC_ADV_SYNC_TRANSFER) |
| static int |
| cmd_sync_transfer(int argc, char **argv) |
| { |
| uint16_t service_data; |
| uint16_t conn_handle; |
| uint16_t sync_handle; |
| int rc; |
| |
| rc = parse_arg_init(argc - 1, argv + 1); |
| if (rc != 0) { |
| return rc; |
| } |
| |
| conn_handle = parse_arg_uint16("conn", &rc); |
| if (rc != 0) { |
| console_printf("invalid 'conn' parameter\n"); |
| return rc; |
| } |
| |
| sync_handle = parse_arg_uint16_dflt("sync_handle", 0, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'sync_handle' parameter\n"); |
| return rc; |
| } |
| |
| service_data = parse_arg_uint16_dflt("service_data", 0, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'service_data' parameter\n"); |
| return rc; |
| } |
| |
| rc = ble_gap_periodic_adv_sync_transfer(sync_handle, conn_handle, |
| service_data); |
| if (rc) { |
| console_printf("Failed to transfer sync (%d)\n", rc); |
| } |
| |
| return rc; |
| } |
| |
| static int |
| cmd_sync_reporting(int argc, char **argv) |
| { |
| uint16_t sync_handle; |
| bool enable; |
| int rc; |
| |
| rc = parse_arg_init(argc - 1, argv + 1); |
| if (rc != 0) { |
| return rc; |
| } |
| |
| sync_handle = parse_arg_uint16_dflt("sync_handle", 0, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'sync_handle' parameter\n"); |
| return rc; |
| } |
| |
| enable = parse_arg_bool_dflt("enabled", 0, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'enabled' parameter\n"); |
| return rc; |
| } |
| |
| rc = ble_gap_periodic_adv_sync_reporting(sync_handle, enable, NULL); |
| if (rc) { |
| console_printf("Failed to %s reporting (%d)\n", |
| enable ? "enable" : "disable", rc); |
| } |
| |
| return rc; |
| } |
| |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| static const struct shell_param sync_transfer_params[] = { |
| {"conn", "connection handle, usage: =<UINT16>"}, |
| {"sync_handle", "sync handle, usage: =[UINT16], default: 0"}, |
| {"service_data", "service data, usage: =[UINT16], default: 0"}, |
| {NULL, NULL} |
| }; |
| |
| static const struct shell_cmd_help sync_transfer_help = { |
| .summary = "start periodic sync transfer procedure with specific parameters", |
| .usage = NULL, |
| .params = sync_transfer_params, |
| }; |
| |
| static const struct shell_param sync_reporting_params[] = { |
| {"sync_handle", "sync handle, usage: =[UINT16], default: 0"}, |
| {"enabled", "toggle reporting, usage: =[0-1], default: 0"}, |
| {NULL, NULL} |
| }; |
| |
| static const struct shell_cmd_help sync_reporting_help = { |
| .summary = "configure periodic advertising sync reporting", |
| .usage = NULL, |
| .params = sync_reporting_params, |
| }; |
| #endif |
| |
| static int |
| cmd_sync_transfer_set_info(int argc, char **argv) |
| { |
| uint16_t service_data; |
| uint16_t conn_handle; |
| uint8_t instance; |
| int rc; |
| |
| rc = parse_arg_init(argc - 1, argv + 1); |
| if (rc != 0) { |
| return rc; |
| } |
| |
| conn_handle = parse_arg_uint16("conn", &rc); |
| if (rc != 0) { |
| console_printf("invalid 'conn' parameter\n"); |
| return rc; |
| } |
| |
| instance = parse_arg_uint8_dflt("instance", 0, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'instance' parameter\n"); |
| return rc; |
| } |
| |
| service_data = parse_arg_uint16_dflt("service_data", 0, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'service_data' parameter\n"); |
| return rc; |
| } |
| |
| rc = ble_gap_periodic_adv_sync_set_info(instance, conn_handle, |
| service_data); |
| if (rc) { |
| console_printf("Failed to transfer sync (%d)\n", rc); |
| } |
| |
| return rc; |
| } |
| |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| static const struct shell_param sync_transfer_set_info_params[] = { |
| {"conn", "connection handle, usage: =<UINT16>"}, |
| {"instance", "advertising instance, usage: =[UINT8], default: 0"}, |
| {"service_data", "service data, usage: =[UINT16], default: 0"}, |
| {NULL, NULL} |
| }; |
| |
| static const struct shell_cmd_help sync_transfer_set_info_help = { |
| .summary = "start periodic sync transfer procedure with specific parameters", |
| .usage = NULL, |
| .params = sync_transfer_set_info_params, |
| }; |
| #endif |
| |
| static int |
| cmd_sync_transfer_receive(int argc, char **argv) |
| { |
| struct ble_gap_periodic_sync_params params = { 0 }; |
| uint16_t conn_handle; |
| bool disable; |
| int rc; |
| |
| rc = parse_arg_init(argc - 1, argv + 1); |
| if (rc != 0) { |
| return rc; |
| } |
| |
| conn_handle = parse_arg_uint16("conn", &rc); |
| if (rc != 0) { |
| console_printf("invalid 'conn' parameter\n"); |
| return rc; |
| } |
| |
| disable = parse_arg_bool_dflt("disable", false, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'disable' parameter\n"); |
| return rc; |
| } |
| |
| if (disable) { |
| rc = ble_gap_periodic_adv_sync_receive(conn_handle, NULL, NULL, NULL); |
| if (rc) { |
| console_printf("Failed to disable sync transfer reception (%d)\n", rc); |
| } |
| |
| return rc; |
| } |
| |
| params.skip = parse_arg_uint16_dflt("skip", 0, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'skip' parameter\n"); |
| return rc; |
| } |
| |
| params.sync_timeout = parse_arg_time_dflt("sync_timeout", 10000, 10, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'sync_timeout' parameter\n"); |
| return rc; |
| } |
| |
| params.reports_disabled = parse_arg_bool_dflt("reports_disabled", false, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'reports_disabled' parameter\n"); |
| return rc; |
| } |
| |
| rc = ble_gap_periodic_adv_sync_receive(conn_handle, ¶ms, btshell_gap_event, |
| NULL); |
| if (rc) { |
| console_printf("Failed to enable sync transfer reception (%d)\n", rc); |
| } |
| |
| return rc; |
| } |
| |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| static const struct shell_param sync_transfer_receive_params[] = { |
| {"conn", "connection handle, usage: =<UINT16>"}, |
| {"disable", "disable transfer reception, usage: =[0-1], default: 0"}, |
| {"skip", "usage: =[0-0x01F3], default: 0x0000"}, |
| {"sync_timeout", "usage: =[0x000A-0x4000], default: 0x000A"}, |
| {"reports_disabled", "disable reports, usage: =[0-1], default: 0"}, |
| {NULL, NULL} |
| }; |
| #endif |
| |
| static const struct shell_cmd_help sync_transfer_receive_help = { |
| .summary = "start/stop periodic sync reception with specific parameters", |
| .usage = NULL, |
| .params = sync_transfer_receive_params, |
| }; |
| #endif |
| |
| static int |
| cmd_sync_terminate(int argc, char **argv) |
| { |
| uint16_t sync_handle; |
| int rc; |
| |
| rc = parse_arg_init(argc - 1, argv + 1); |
| if (rc != 0) { |
| return rc; |
| } |
| |
| sync_handle = parse_arg_uint16_dflt("sync_handle", 0, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'sync_handle' parameter\n"); |
| return rc; |
| } |
| |
| rc = ble_gap_periodic_adv_sync_terminate(sync_handle); |
| if (rc) { |
| console_printf("Failed to terminate sync (%d)\n", rc); |
| } |
| |
| return rc; |
| } |
| |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| static const struct shell_param sync_terminate_params[] = { |
| {"sync_handle", "usage: =[UINT16], default: 0"}, |
| {NULL, NULL} |
| }; |
| |
| static const struct shell_cmd_help sync_terminate_help = { |
| .summary = "terminate periodic sync", |
| .usage = NULL, |
| .params = sync_terminate_params, |
| }; |
| #endif |
| |
| static int |
| cmd_sync_stats(int argc, char **argv) |
| { |
| uint16_t sync_handle; |
| int rc; |
| |
| rc = parse_arg_init(argc - 1, argv + 1); |
| if (rc != 0) { |
| return rc; |
| } |
| |
| sync_handle = parse_arg_uint16_dflt("sync_handle", 0, &rc); |
| if (rc != 0) { |
| console_printf("invalid 'sync_handle' parameter\n"); |
| return rc; |
| } |
| |
| btshell_sync_stats(sync_handle); |
| |
| return 0; |
| } |
| |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| static const struct shell_param sync_stats_params[] = { |
| {"sync_handle", "usage: =[UINT16], default: 0"}, |
| {NULL, NULL} |
| }; |
| |
| static const struct shell_cmd_help sync_stats_help = { |
| .summary = "show sync stats", |
| .usage = NULL, |
| .params = sync_stats_params, |
| }; |
| #endif |
| #endif |
| |
| #if MYNEWT_VAL(BLE_ISO_BROADCAST_SOURCE) |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| static const struct shell_param leaudio_base_add_params[] = { |
| {"adv_instance", "Advertising instance, usage: =<UINT8>"}, |
| { "presentation_delay", "usage: =<UINT32>" }, |
| |
| { NULL, NULL} |
| }; |
| |
| static const struct shell_cmd_help leaudio_base_add_help = { |
| .summary = "Add BASE configuration for broadcast", |
| .usage = NULL, |
| .params = leaudio_base_add_params, |
| }; |
| |
| static const struct shell_param leaudio_big_sub_add_params[] = { |
| {"adv_instance", "Advertising instance, usage: =<UINT8>"}, |
| { "codec_fmt", "usage: <UINT8>" }, |
| { "company_id", "usage: =<UINT16>" }, |
| { "vendor_spec", "usage: =<UINT16>" }, |
| { "codec_spec_config", "usage: =[XX:XX...]" }, |
| { "metadata", "usage: =[XX:XX...]" }, |
| |
| { NULL, NULL} |
| }; |
| |
| static const struct shell_cmd_help leaudio_big_sub_add_help = { |
| .summary = "Add BIG subgroup configuration for broadcast", |
| .usage = NULL, |
| .params = leaudio_big_sub_add_params, |
| }; |
| |
| static const struct shell_param leaudio_bis_add_params[] = { |
| {"adv_instance", "Advertising instance, usage: =<UINT8>"}, |
| { "codec_spec_config", "usage: =[XX:XX...]" }, |
| |
| { NULL, NULL} |
| }; |
| |
| static const struct shell_cmd_help leaudio_bis_add_help = { |
| .summary = "Add BIS configuration for broadcast announcements to last " |
| "added BIG subgroup", |
| .usage = NULL, |
| .params = leaudio_bis_add_params, |
| }; |
| |
| static const struct shell_param leaudio_broadcast_create_params[] = { |
| {"adv_instance", "Advertising instance, usage: =<UINT8>"}, |
| {"ext_interval_min", "usage: =[0-UINT16_MAX], default: 0"}, |
| {"ext_interval_max", "usage: =[0-UINT16_MAX], default: 0"}, |
| {"per_interval_min", "usage: =[0-UINT16_MAX], default: 0"}, |
| {"per_interval_max", "usage: =[0-UINT16_MAX], default: 0"}, |
| |
| {"name", "usage: =[string]"}, |
| |
| {"sdu_interval", "SDU interval, in us, usage: =<UINT32>"}, |
| {"max_sdu", "max SDU size, in octets, usage: =<UINT16>"}, |
| {"max_latency", "max transport latency, in ms, usage: =<UINT16>"}, |
| {"rtn", "RTN, usage: =<UINT8>"}, |
| {"phy", "PHY, usage: =<UINT8>"}, |
| {"packing", "packing, optional, true if not given, usage: =<UINT8>"}, |
| {"framing", "framing, optional, false if not given, usage: =<UINT8>"}, |
| {"encryption", "optional, encryption, usage: =<UINT8>"}, |
| {"broadcast_code", "optional, obligatory if encryption is enabled, " |
| "usage: =<XX:XX:...>, len=16 octets"}, |
| {"extra_data", "usage: =[XX:XX...]"}, |
| |
| { NULL, NULL} |
| }; |
| |
| static const struct shell_cmd_help leaudio_broadcast_create_help = { |
| .summary = "Add BIS configuration for broadcast to last added BIG " |
| "subgroup", |
| .usage = NULL, |
| .params = leaudio_broadcast_create_params, |
| }; |
| |
| static const struct shell_param leaudio_broadcast_destroy_params[] = { |
| {"adv_instance", "Advertising instance, usage: =<UINT8>"}, |
| |
| { NULL, NULL} |
| }; |
| |
| static const struct shell_cmd_help leaudio_broadcast_destroy_help = { |
| .summary = "Destroy BASE", |
| .usage = NULL, |
| |
| .params = leaudio_broadcast_destroy_params, |
| }; |
| |
| static const struct shell_param leaudio_broadcast_update_params[] = { |
| {"adv_instance", "Advertising instance, usage: =<UINT8>"}, |
| {"name", "usage: =[string]"}, |
| {"extra_data_len", "usage: =<UINT8>"}, |
| {"extra_data", "usage: =[XX:XX...]"}, |
| |
| { NULL, NULL} |
| }; |
| |
| static const struct shell_cmd_help leaudio_broadcast_update_help = { |
| .summary = "Update broadcast", |
| .usage = NULL, |
| |
| .params = leaudio_broadcast_update_params, |
| }; |
| |
| static const struct shell_param leaudio_broadcast_start_params[] = { |
| {"adv_instance", "Advertising instance, usage: =<UINT8>"}, |
| |
| { NULL, NULL} |
| }; |
| |
| static const struct shell_cmd_help leaudio_broadcast_start_help = { |
| .summary = "Start broadcast", |
| .usage = NULL, |
| |
| .params = leaudio_broadcast_start_params, |
| }; |
| |
| static const struct shell_param leaudio_broadcast_stop_params[] = { |
| {"adv_instance", "Advertising instance, usage: =<UINT8>"}, |
| |
| { NULL, NULL} |
| }; |
| |
| static const struct shell_cmd_help leaudio_broadcast_stop_help = { |
| .summary = "Stop broadcast", |
| .usage = NULL, |
| |
| .params = leaudio_broadcast_stop_params, |
| }; |
| #endif |
| #endif |
| |
| static const struct shell_cmd btshell_commands[] = { |
| #if MYNEWT_VAL(BLE_EXT_ADV) |
| { |
| .sc_cmd = "advertise-configure", |
| .sc_cmd_func = cmd_advertise_configure, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &advertise_configure_help, |
| #endif |
| }, |
| { |
| .sc_cmd = "advertise-set-addr", |
| .sc_cmd_func = cmd_advertise_set_addr, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &advertise_set_addr_help, |
| #endif |
| }, |
| { |
| .sc_cmd = "advertise-set-adv-data", |
| .sc_cmd_func = cmd_set_adv_data, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &set_adv_data_help, |
| #endif |
| }, |
| { |
| .sc_cmd = "advertise-set-scan-rsp", |
| .sc_cmd_func = cmd_set_scan_rsp, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &set_scan_rsp_help, |
| #endif |
| }, |
| { |
| .sc_cmd = "advertise-start", |
| .sc_cmd_func = cmd_advertise_start, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &advertise_start_help, |
| #endif |
| }, |
| { |
| .sc_cmd = "advertise-stop", |
| .sc_cmd_func = cmd_advertise_stop, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &advertise_stop_help, |
| #endif |
| }, |
| { |
| .sc_cmd = "advertise-remove", |
| .sc_cmd_func = cmd_advertise_remove, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &advertise_remove_help, |
| #endif |
| }, |
| #else |
| { |
| .sc_cmd = "advertise", |
| .sc_cmd_func = cmd_advertise, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &advertise_help, |
| #endif |
| }, |
| #endif |
| { |
| .sc_cmd = "connect", |
| .sc_cmd_func = cmd_connect, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &connect_help, |
| #endif |
| }, |
| { |
| .sc_cmd = "disconnect", |
| .sc_cmd_func = cmd_disconnect, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &disconnect_help, |
| #endif |
| }, |
| { |
| .sc_cmd = "show-addr", |
| .sc_cmd_func = cmd_show_addr, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &gatt_show_addr_help, |
| #endif |
| }, |
| { |
| .sc_cmd = "show-conn", |
| .sc_cmd_func = cmd_show_conn, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &gatt_show_conn_help, |
| #endif |
| }, |
| { |
| .sc_cmd = "set-scan-opts", |
| .sc_cmd_func = cmd_set_scan_opts, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &set_scan_opts_help, |
| #endif |
| }, |
| { |
| .sc_cmd = "scan", |
| .sc_cmd_func = cmd_scan, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &scan_help, |
| #endif |
| }, |
| { |
| .sc_cmd = "set", |
| .sc_cmd_func = cmd_set, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &set_help, |
| #endif |
| }, |
| #if !MYNEWT_VAL(BLE_EXT_ADV) |
| { |
| .sc_cmd = "set-adv-data", |
| .sc_cmd_func = cmd_set_adv_data, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &set_adv_data_help, |
| #endif |
| }, |
| { |
| .sc_cmd = "set-scan-rsp", |
| .sc_cmd_func = cmd_set_scan_rsp, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &set_scan_rsp_help, |
| #endif |
| }, |
| #endif |
| { |
| .sc_cmd = "set-priv-mode", |
| .sc_cmd_func = cmd_set_priv_mode, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &set_priv_mode_help, |
| #endif |
| }, |
| { |
| .sc_cmd = "white-list", |
| .sc_cmd_func = cmd_white_list, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &white_list_help, |
| #endif |
| }, |
| { |
| .sc_cmd = "conn-rssi", |
| .sc_cmd_func = cmd_conn_rssi, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &conn_rssi_help, |
| #endif |
| }, |
| { |
| .sc_cmd = "conn-update-params", |
| .sc_cmd_func = cmd_conn_update_params, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &conn_update_params_help, |
| #endif |
| }, |
| { |
| .sc_cmd = "conn-datalen", |
| .sc_cmd_func = cmd_conn_datalen, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &conn_datalen_help, |
| #endif |
| }, |
| { |
| .sc_cmd = "gatt-discover-characteristic", |
| .sc_cmd_func = cmd_gatt_discover_characteristic, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &gatt_discover_characteristic_help, |
| #endif |
| }, |
| { |
| .sc_cmd = "gatt-discover-descriptor", |
| .sc_cmd_func = cmd_gatt_discover_descriptor, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &gatt_discover_descriptor_help, |
| #endif |
| }, |
| { |
| .sc_cmd = "gatt-discover-service", |
| .sc_cmd_func = cmd_gatt_discover_service, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &gatt_discover_service_help, |
| #endif |
| }, |
| { |
| .sc_cmd = "gatt-discover-full", |
| .sc_cmd_func = cmd_gatt_discover_full, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &gatt_discover_full_help, |
| #endif |
| }, |
| { |
| .sc_cmd = "gatt-find-included-services", |
| .sc_cmd_func = cmd_gatt_find_included_services, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &gatt_find_included_services_help, |
| #endif |
| }, |
| { |
| .sc_cmd = "gatt-exchange-mtu", |
| .sc_cmd_func = cmd_gatt_exchange_mtu, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &gatt_exchange_mtu_help, |
| #endif |
| }, |
| { |
| .sc_cmd = "gatt-read", |
| .sc_cmd_func = cmd_gatt_read, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &gatt_read_help, |
| #endif |
| }, |
| { |
| .sc_cmd = "gatt-notify", |
| .sc_cmd_func = cmd_gatt_notify, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &gatt_notify_help, |
| #endif |
| }, |
| { |
| .sc_cmd = "gatt-service-changed", |
| .sc_cmd_func = cmd_gatt_service_changed, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &gatt_service_changed_help, |
| #endif |
| }, |
| { |
| .sc_cmd = "gatt-service-visibility", |
| .sc_cmd_func = cmd_gatt_service_visibility, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &gatt_service_visibility_help, |
| #endif |
| }, |
| { |
| .sc_cmd = "gatt-show", |
| .sc_cmd_func = cmd_gatt_show, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &gatt_show_help, |
| #endif |
| }, |
| { |
| .sc_cmd = "gatt-show-local", |
| .sc_cmd_func = cmd_gatt_show_local, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &gatt_show_local_help, |
| #endif |
| }, |
| { |
| .sc_cmd = "gatt-write", |
| .sc_cmd_func = cmd_gatt_write, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &gatt_write_help, |
| #endif |
| }, |
| { |
| .sc_cmd = "gatt-enqueue-notif", |
| .sc_cmd_func = cmd_gatt_enqueue_notif, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &gatt_enqueue_notif_help, |
| #endif |
| }, |
| { |
| .sc_cmd = "gatt-send-queued-notif", |
| .sc_cmd_func = cmd_gatt_send_pending_notif, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &gatt_send_pending_notif_help, |
| #endif |
| }, |
| { |
| .sc_cmd = "gatt-clear-queued-notif", |
| .sc_cmd_func = cmd_gatt_clear_pending_notif, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &gatt_clear_pending_notif_help, |
| #endif |
| }, |
| #if MYNEWT_VAL(BLE_L2CAP_COC_MAX_NUM) |
| { |
| .sc_cmd = "l2cap-update", |
| .sc_cmd_func = cmd_l2cap_update, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &l2cap_update_help, |
| #endif |
| }, |
| { |
| .sc_cmd = "l2cap-create-server", |
| .sc_cmd_func = cmd_l2cap_create_server, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &l2cap_create_server_help, |
| #endif |
| }, |
| { |
| .sc_cmd = "l2cap-connect", |
| .sc_cmd_func = cmd_l2cap_connect, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &l2cap_connect_help, |
| #endif |
| }, |
| { |
| .sc_cmd = "l2cap-reconfig", |
| .sc_cmd_func = cmd_l2cap_reconfig, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &l2cap_reconfig_help, |
| #endif |
| }, |
| { |
| .sc_cmd = "l2cap-disconnect", |
| .sc_cmd_func = cmd_l2cap_disconnect, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &l2cap_disconnect_help, |
| #endif |
| }, |
| { |
| .sc_cmd = "l2cap-send", |
| .sc_cmd_func = cmd_l2cap_send, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &l2cap_send_help, |
| #endif |
| }, |
| { |
| .sc_cmd = "l2cap-show-coc", |
| .sc_cmd_func = cmd_l2cap_show_coc, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &l2cap_show_coc_help, |
| #endif |
| }, |
| #endif |
| { |
| .sc_cmd = "keystore-add", |
| .sc_cmd_func = cmd_keystore_add, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &keystore_add_help, |
| #endif |
| }, |
| { |
| .sc_cmd = "keystore-del", |
| .sc_cmd_func = cmd_keystore_del, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &keystore_del_help, |
| #endif |
| }, |
| { |
| .sc_cmd = "keystore-show", |
| .sc_cmd_func = cmd_keystore_show, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &keystore_show_help, |
| #endif |
| }, |
| #if NIMBLE_BLE_SM |
| { |
| .sc_cmd = "show-oob-sc", |
| .sc_cmd_func = cmd_show_oob_sc, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = NULL, |
| #endif |
| }, |
| { |
| .sc_cmd = "auth-passkey", |
| .sc_cmd_func = cmd_auth_passkey, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &auth_passkey_help, |
| #endif |
| }, |
| { |
| .sc_cmd = "security-pair", |
| .sc_cmd_func = cmd_security_pair, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &security_pair_help, |
| #endif |
| }, |
| { |
| .sc_cmd = "security-unpair", |
| .sc_cmd_func = cmd_security_unpair, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &security_unpair_help, |
| #endif |
| }, |
| { |
| .sc_cmd = "security-start", |
| .sc_cmd_func = cmd_security_start, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &security_start_help, |
| #endif |
| }, |
| { |
| .sc_cmd = "security-encryption", |
| .sc_cmd_func = cmd_security_encryption, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &security_encryption_help, |
| #endif |
| }, |
| { |
| .sc_cmd = "security-set-data", |
| .sc_cmd_func = cmd_security_set_data, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &security_set_data_help, |
| #endif |
| }, |
| #endif |
| { |
| .sc_cmd = "test-tx", |
| .sc_cmd_func = cmd_test_tx, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &test_tx_help, |
| #endif |
| }, |
| { |
| .sc_cmd = "phy-set", |
| .sc_cmd_func = cmd_phy_set, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &phy_set_help, |
| #endif |
| }, |
| { |
| .sc_cmd = "phy-set-default", |
| .sc_cmd_func = cmd_phy_set_default, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &phy_set_default_help, |
| #endif |
| }, |
| { |
| .sc_cmd = "phy-read", |
| .sc_cmd_func = cmd_phy_read, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &phy_read_help, |
| #endif |
| }, |
| { |
| .sc_cmd = "host-enable", |
| .sc_cmd_func = cmd_host_enable, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &host_enable_help, |
| #endif |
| }, |
| { |
| .sc_cmd = "host-disable", |
| .sc_cmd_func = cmd_host_disable, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &host_disable_help, |
| #endif |
| }, |
| #if MYNEWT_VAL(BLE_PERIODIC_ADV) |
| { |
| .sc_cmd = "periodic-configure", |
| .sc_cmd_func = cmd_periodic_configure, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &periodic_configure_help, |
| #endif |
| }, |
| { |
| .sc_cmd = "periodic-set-adv-data", |
| .sc_cmd_func = cmd_periodic_set_adv_data, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &set_adv_data_help, |
| #endif |
| }, |
| { |
| .sc_cmd = "periodic-start", |
| .sc_cmd_func = cmd_periodic_start, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &periodic_start_help, |
| #endif |
| }, |
| { |
| .sc_cmd = "periodic-stop", |
| .sc_cmd_func = cmd_periodic_stop, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &periodic_stop_help, |
| #endif |
| }, |
| { |
| .sc_cmd = "sync-create", |
| .sc_cmd_func = cmd_sync_create, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &sync_create_help, |
| #endif |
| }, |
| { |
| .sc_cmd = "sync-terminate", |
| .sc_cmd_func = cmd_sync_terminate, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &sync_terminate_help, |
| #endif |
| }, |
| { |
| .sc_cmd = "sync-stats", |
| .sc_cmd_func = cmd_sync_stats, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &sync_stats_help, |
| #endif |
| }, |
| #if MYNEWT_VAL(BLE_PERIODIC_ADV_SYNC_TRANSFER) |
| { |
| .sc_cmd = "sync-transfer", |
| .sc_cmd_func = cmd_sync_transfer, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &sync_transfer_help, |
| #endif |
| }, |
| { |
| .sc_cmd = "sync-transfer-set-info", |
| .sc_cmd_func = cmd_sync_transfer_set_info, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &sync_transfer_set_info_help, |
| #endif |
| }, |
| { |
| .sc_cmd = "sync-transfer-receive", |
| .sc_cmd_func = cmd_sync_transfer_receive, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &sync_transfer_receive_help, |
| #endif |
| }, |
| { |
| .sc_cmd = "sync-reporting", |
| .sc_cmd_func = cmd_sync_reporting, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &sync_reporting_help, |
| #endif |
| }, |
| #endif |
| #endif |
| #if MYNEWT_VAL(BLE_ISO_BROADCAST_SOURCE) |
| { |
| .sc_cmd = "base_add", |
| .sc_cmd_func = cmd_leaudio_base_add, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &leaudio_base_add_help, |
| #endif |
| }, |
| { |
| .sc_cmd = "big_sub_add", |
| .sc_cmd_func = cmd_leaudio_big_sub_add, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &leaudio_big_sub_add_help, |
| #endif |
| }, |
| { |
| .sc_cmd = "bis_add", |
| .sc_cmd_func = cmd_leaudio_bis_add, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &leaudio_bis_add_help, |
| #endif |
| }, |
| { |
| .sc_cmd = "broadcast_create", |
| .sc_cmd_func = cmd_leaudio_broadcast_create, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &leaudio_broadcast_create_help, |
| #endif |
| }, |
| { |
| .sc_cmd = "broadcast_destroy", |
| .sc_cmd_func = cmd_leaudio_broadcast_destroy, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &leaudio_broadcast_destroy_help, |
| #endif |
| }, |
| { |
| .sc_cmd = "broadcast_update", |
| .sc_cmd_func = cmd_leaudio_broadcast_update, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &leaudio_broadcast_update_help, |
| #endif |
| }, |
| { |
| .sc_cmd = "broadcast_start", |
| .sc_cmd_func = cmd_leaudio_broadcast_start, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &leaudio_broadcast_start_help, |
| #endif |
| }, |
| { |
| .sc_cmd = "broadcast_stop", |
| .sc_cmd_func = cmd_leaudio_broadcast_stop, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &leaudio_broadcast_stop_help, |
| #endif |
| }, |
| #endif /* BLE_ISO_BROADCAST_SOURCE */ |
| #if MYNEWT_VAL(BLE_ISO) |
| #if MYNEWT_VAL(BLE_ISO_BROADCAST_SOURCE) |
| { |
| .sc_cmd = "big-create", |
| .sc_cmd_func = cmd_iso_big_create, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &cmd_iso_big_create_help, |
| #endif |
| }, |
| { |
| .sc_cmd = "big-terminate", |
| .sc_cmd_func = cmd_iso_big_terminate, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &cmd_iso_big_terminate_help, |
| #endif |
| }, |
| #endif /* BLE_ISO_BROADCAST_SOURCE */ |
| #if MYNEWT_VAL(BLE_ISO_BROADCAST_SINK) |
| { |
| .sc_cmd = "big-sync-create", |
| .sc_cmd_func = cmd_iso_big_sync_create, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &cmd_iso_big_sync_create_help, |
| #endif |
| }, |
| { |
| .sc_cmd = "big-sync-terminate", |
| .sc_cmd_func = cmd_iso_big_sync_terminate, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &cmd_iso_big_sync_terminate_help, |
| #endif |
| }, |
| #endif /* BLE_ISO_BROADCAST_SINK */ |
| { |
| .sc_cmd = "iso-data-path-setup", |
| .sc_cmd_func = cmd_iso_data_path_setup, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &cmd_iso_data_path_setup_help, |
| #endif |
| }, |
| { |
| .sc_cmd = "iso-data-path-remove", |
| .sc_cmd_func = cmd_iso_data_path_remove, |
| #if MYNEWT_VAL(SHELL_CMD_HELP) |
| .help = &cmd_iso_data_path_remove_help, |
| #endif |
| }, |
| #endif /* BLE_ISO */ |
| { 0 }, |
| }; |
| |
| |
| void |
| cmd_init(void) |
| { |
| shell_register(BTSHELL_MODULE, btshell_commands); |
| shell_register_default_module(BTSHELL_MODULE); |
| } |