#include #include #include #include #include "bt_settings.h" static const char* bt_cli_address_types[] = { "Public Device Address", "Random Device Address", "Public Identity Address", "Random (Static) Identity Address", }; static void bt_cli_command_hci_info(Cli* cli, string_t args, void* context) { string_t buffer; string_init(buffer); furi_hal_bt_dump_state(buffer); printf("%s", string_get_cstr(buffer)); string_clear(buffer); } static void bt_cli_command_carrier_tx(Cli* cli, string_t args, void* context) { int channel = 0; int power = 0; do { if(!args_read_int_and_trim(args, &channel) && (channel < 0 || channel > 39)) { printf("Incorrect or missing channel, expected int 0-39"); break; } if(!args_read_int_and_trim(args, &power) && (power < 0 || power > 6)) { printf("Incorrect or missing power, expected int 0-6"); break; } furi_hal_bt_stop_advertising(); printf("Transmitting carrier at %d channel at %d dB power\r\n", channel, power); printf("Press CTRL+C to stop\r\n"); furi_hal_bt_start_tone_tx(channel, 0x19 + power); while(!cli_cmd_interrupt_received(cli)) { osDelay(250); } furi_hal_bt_stop_tone_tx(); } while(false); } static void bt_cli_command_carrier_rx(Cli* cli, string_t args, void* context) { int channel = 0; do { if(!args_read_int_and_trim(args, &channel) && (channel < 0 || channel > 39)) { printf("Incorrect or missing channel, expected int 0-39"); break; } furi_hal_bt_stop_advertising(); printf("Receiving carrier at %d channel\r\n", channel); printf("Press CTRL+C to stop\r\n"); furi_hal_bt_start_packet_rx(channel, 1); while(!cli_cmd_interrupt_received(cli)) { osDelay(250); printf("RSSI: %6.1f dB\r", furi_hal_bt_get_rssi()); fflush(stdout); } furi_hal_bt_stop_packet_test(); } while(false); } static void bt_cli_command_packet_tx(Cli* cli, string_t args, void* context) { int channel = 0; int pattern = 0; int datarate = 1; do { if(!args_read_int_and_trim(args, &channel) && (channel < 0 || channel > 39)) { printf("Incorrect or missing channel, expected int 0-39"); break; } if(!args_read_int_and_trim(args, &pattern) && (pattern < 0 || pattern > 5)) { printf("Incorrect or missing pattern, expected int 0-5 \r\n"); printf("0 - Pseudo-Random bit sequence 9\r\n"); printf("1 - Pattern of alternating bits '11110000'\r\n"); printf("2 - Pattern of alternating bits '10101010'\r\n"); printf("3 - Pseudo-Random bit sequence 15\r\n"); printf("4 - Pattern of All '1' bits\r\n"); printf("5 - Pattern of All '0' bits\r\n"); break; } if(!args_read_int_and_trim(args, &datarate) && (datarate < 1 || datarate > 2)) { printf("Incorrect or missing datarate, expected int 1-2"); break; } furi_hal_bt_stop_advertising(); printf( "Transmitting %d pattern packet at %d channel at %d M datarate\r\n", pattern, channel, datarate); printf("Press CTRL+C to stop\r\n"); furi_hal_bt_start_packet_tx(channel, pattern, datarate); while(!cli_cmd_interrupt_received(cli)) { osDelay(250); } furi_hal_bt_stop_packet_test(); printf("Transmitted %lu packets", furi_hal_bt_get_transmitted_packets()); } while(false); } static void bt_cli_command_packet_rx(Cli* cli, string_t args, void* context) { int channel = 0; int datarate = 1; do { if(!args_read_int_and_trim(args, &channel) && (channel < 0 || channel > 39)) { printf("Incorrect or missing channel, expected int 0-39"); break; } if(!args_read_int_and_trim(args, &datarate) && (datarate < 1 || datarate > 2)) { printf("Incorrect or missing datarate, expected int 1-2"); break; } furi_hal_bt_stop_advertising(); printf("Receiving packets at %d channel at %d M datarate\r\n", channel, datarate); printf("Press CTRL+C to stop\r\n"); furi_hal_bt_start_packet_rx(channel, datarate); float rssi_raw = 0; while(!cli_cmd_interrupt_received(cli)) { osDelay(250); rssi_raw = furi_hal_bt_get_rssi(); printf("RSSI: %03.1f dB\r", rssi_raw); fflush(stdout); } uint16_t packets_received = furi_hal_bt_stop_packet_test(); printf("Received %hu packets", packets_received); } while(false); } static void bt_cli_scan_callback(GapAddress address, void* context) { furi_assert(context); osMessageQueueId_t queue = context; osMessageQueuePut(queue, &address, NULL, 250); } static void bt_cli_command_scan(Cli* cli, string_t args, void* context) { osMessageQueueId_t queue = osMessageQueueNew(20, sizeof(GapAddress), NULL); furi_hal_bt_start_scan(bt_cli_scan_callback, queue); GapAddress address = {}; bool exit = false; while(!exit) { if(osMessageQueueGet(queue, &address, NULL, 250) == osOK) { if(address.type < sizeof(bt_cli_address_types)) { printf("Found new device. Type: %s, MAC: ", bt_cli_address_types[address.type]); for(uint8_t i = 0; i < sizeof(address.mac) - 1; i++) { printf("%02X:", address.mac[i]); } printf("%02X\r\n", address.mac[sizeof(address.mac) - 1]); } } exit = cli_cmd_interrupt_received(cli); } furi_hal_bt_stop_scan(); osMessageQueueDelete(queue); } static void bt_cli_print_usage() { printf("Usage:\r\n"); printf("bt \r\n"); printf("Cmd list:\r\n"); printf("\thci_info\t - HCI info\r\n"); if(furi_hal_rtc_is_flag_set(FuriHalRtcFlagDebug) && furi_hal_bt_get_radio_stack() == FuriHalBtStackHciLayer) { printf("\ttx_carrier \t - start tx carrier test\r\n"); printf("\trx_carrier \t - start rx carrier test\r\n"); printf("\ttx_pt \t - start tx packet test\r\n"); printf("\trx_pt \t - start rx packer test\r\n"); printf("\tscan\t - start scanner\r\n"); } } static void bt_cli(Cli* cli, string_t args, void* context) { furi_record_open("bt"); string_t cmd; string_init(cmd); BtSettings bt_settings; bt_settings_load(&bt_settings); do { if(!args_read_string_and_trim(args, cmd)) { bt_cli_print_usage(); break; } if(string_cmp_str(cmd, "hci_info") == 0) { bt_cli_command_hci_info(cli, args, NULL); break; } if(furi_hal_rtc_is_flag_set(FuriHalRtcFlagDebug) && furi_hal_bt_get_radio_stack() == FuriHalBtStackHciLayer) { if(string_cmp_str(cmd, "carrier_tx") == 0) { bt_cli_command_carrier_tx(cli, args, NULL); break; } if(string_cmp_str(cmd, "carrier_rx") == 0) { bt_cli_command_carrier_rx(cli, args, NULL); break; } if(string_cmp_str(cmd, "packet_tx") == 0) { bt_cli_command_packet_tx(cli, args, NULL); break; } if(string_cmp_str(cmd, "packet_rx") == 0) { bt_cli_command_packet_rx(cli, args, NULL); break; } if(string_cmp_str(cmd, "scan") == 0) { bt_cli_command_scan(cli, args, NULL); break; } } bt_cli_print_usage(); } while(false); if(bt_settings.enabled) { furi_hal_bt_start_advertising(); } string_clear(cmd); furi_record_close("bt"); } void bt_on_system_start() { #ifdef SRV_CLI Cli* cli = furi_record_open("cli"); cli_add_command(cli, "bt", CliCommandFlagDefault, bt_cli, NULL); furi_record_close("cli"); #else UNUSED(bt_cli); #endif }