11b464e99SMatthias Ringwald /* 21b464e99SMatthias Ringwald * Copyright (C) 2019 BlueKitchen GmbH 31b464e99SMatthias Ringwald * 41b464e99SMatthias Ringwald * Redistribution and use in source and binary forms, with or without 51b464e99SMatthias Ringwald * modification, are permitted provided that the following conditions 61b464e99SMatthias Ringwald * are met: 71b464e99SMatthias Ringwald * 81b464e99SMatthias Ringwald * 1. Redistributions of source code must retain the above copyright 91b464e99SMatthias Ringwald * notice, this list of conditions and the following disclaimer. 101b464e99SMatthias Ringwald * 2. Redistributions in binary form must reproduce the above copyright 111b464e99SMatthias Ringwald * notice, this list of conditions and the following disclaimer in the 121b464e99SMatthias Ringwald * documentation and/or other materials provided with the distribution. 131b464e99SMatthias Ringwald * 3. Neither the name of the copyright holders nor the names of 141b464e99SMatthias Ringwald * contributors may be used to endorse or promote products derived 151b464e99SMatthias Ringwald * from this software without specific prior written permission. 161b464e99SMatthias Ringwald * 4. Any redistribution, use, or modification is done solely for 171b464e99SMatthias Ringwald * personal benefit and not for any commercial purpose or for 181b464e99SMatthias Ringwald * monetary gain. 191b464e99SMatthias Ringwald * 201b464e99SMatthias Ringwald * THIS SOFTWARE IS PROVIDED BY BLUEKITCHEN GMBH AND CONTRIBUTORS 211b464e99SMatthias Ringwald * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT 221b464e99SMatthias Ringwald * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS 231b464e99SMatthias Ringwald * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL MATTHIAS 241b464e99SMatthias Ringwald * RINGWALD OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, 251b464e99SMatthias Ringwald * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, 261b464e99SMatthias Ringwald * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS 271b464e99SMatthias Ringwald * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED 281b464e99SMatthias Ringwald * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, 291b464e99SMatthias Ringwald * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF 301b464e99SMatthias Ringwald * THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 311b464e99SMatthias Ringwald * SUCH DAMAGE. 321b464e99SMatthias Ringwald * 331b464e99SMatthias Ringwald * Please inquire about commercial licensing options at 341b464e99SMatthias Ringwald * [email protected] 351b464e99SMatthias Ringwald * 361b464e99SMatthias Ringwald */ 371b464e99SMatthias Ringwald 381b464e99SMatthias Ringwald #define __BTSTACK_FILE__ "main.c" 391b464e99SMatthias Ringwald 401b464e99SMatthias Ringwald // ***************************************************************************** 411b464e99SMatthias Ringwald // 421b464e99SMatthias Ringwald // minimal setup for HCI code 431b464e99SMatthias Ringwald // 441b464e99SMatthias Ringwald // ***************************************************************************** 451b464e99SMatthias Ringwald 461b464e99SMatthias Ringwald #include <QCoreApplication> 471b464e99SMatthias Ringwald 481b464e99SMatthias Ringwald #include <stdint.h> 491b464e99SMatthias Ringwald #include <stdio.h> 501b464e99SMatthias Ringwald #include <stdlib.h> 511b464e99SMatthias Ringwald #include <string.h> 521b464e99SMatthias Ringwald #include <signal.h> 531b464e99SMatthias Ringwald 541b464e99SMatthias Ringwald #include "btstack_config.h" 551b464e99SMatthias Ringwald 561b464e99SMatthias Ringwald #include "bluetooth_company_id.h" 571b464e99SMatthias Ringwald #include "btstack_debug.h" 581b464e99SMatthias Ringwald #include "btstack_event.h" 591b464e99SMatthias Ringwald #include "ble/le_device_db_tlv.h" 601b464e99SMatthias Ringwald #include "classic/btstack_link_key_db_tlv.h" 611b464e99SMatthias Ringwald #include "btstack_memory.h" 621b464e99SMatthias Ringwald #include "btstack_run_loop.h" 631b464e99SMatthias Ringwald #include "btstack_run_loop_qt.h" 641b464e99SMatthias Ringwald #include "hal_led.h" 651b464e99SMatthias Ringwald #include "hci.h" 661b464e99SMatthias Ringwald #include "hci_dump.h" 677435ec7bSMatthias Ringwald #include "hci_dump_posix_fs.h" 68c8dfe071SMatthias Ringwald #include "hci_transport.h" 69c8dfe071SMatthias Ringwald #include "hci_transport_usb.h" 701b464e99SMatthias Ringwald #include "btstack_stdin.h" 711b464e99SMatthias Ringwald #include "btstack_audio.h" 721b464e99SMatthias Ringwald #include "btstack_tlv_posix.h" 7384ede529SMatthias Ringwald #include "btstack_uart_block.h" 741b464e99SMatthias Ringwald 75208aaac2SMatthias Ringwald #ifdef Q_OS_WIN 76*00f9da3aSMatthias Ringwald #include "btstack_stdin_windows.h" 77*00f9da3aSMatthias Ringwald #else 78*00f9da3aSMatthias Ringwald #include <signal.h> 79*00f9da3aSMatthias Ringwald #include "btstack_signal.h" 80*00f9da3aSMatthias Ringwald #endif 81*00f9da3aSMatthias Ringwald 82*00f9da3aSMatthias Ringwald #ifdef Q_OS_WIN 83208aaac2SMatthias Ringwald #define TLV_DB_PATH_PREFIX "btstack" 84208aaac2SMatthias Ringwald #else 851b464e99SMatthias Ringwald #define TLV_DB_PATH_PREFIX "/tmp/btstack_" 86208aaac2SMatthias Ringwald #endif 87208aaac2SMatthias Ringwald 881b464e99SMatthias Ringwald #define TLV_DB_PATH_POSTFIX ".tlv" 891b464e99SMatthias Ringwald static char tlv_db_path[100]; 901b464e99SMatthias Ringwald static const btstack_tlv_t * tlv_impl; 911b464e99SMatthias Ringwald static btstack_tlv_posix_t tlv_context; 921b464e99SMatthias Ringwald static bd_addr_t local_addr; 93*00f9da3aSMatthias Ringwald // shutdown 94*00f9da3aSMatthias Ringwald static bool shutdown_triggered; 951b464e99SMatthias Ringwald 961b464e99SMatthias Ringwald extern "C" int btstack_main(int argc, const char * argv[]); 971b464e99SMatthias Ringwald 981b464e99SMatthias Ringwald static const uint8_t read_static_address_command_complete_prefix[] = { 0x0e, 0x1b, 0x01, 0x09, 0xfc }; 991b464e99SMatthias Ringwald 1001b464e99SMatthias Ringwald static bd_addr_t static_address; 1011b464e99SMatthias Ringwald static int using_static_address; 1021b464e99SMatthias Ringwald 1031b464e99SMatthias Ringwald static btstack_packet_callback_registration_t hci_event_callback_registration; 1041b464e99SMatthias Ringwald 1051b464e99SMatthias Ringwald static void local_version_information_handler(uint8_t * packet){ 1061b464e99SMatthias Ringwald printf("Local version information:\n"); 1071b464e99SMatthias Ringwald uint16_t hci_version = packet[6]; 1081b464e99SMatthias Ringwald uint16_t hci_revision = little_endian_read_16(packet, 7); 1091b464e99SMatthias Ringwald uint16_t lmp_version = packet[9]; 1101b464e99SMatthias Ringwald uint16_t manufacturer = little_endian_read_16(packet, 10); 1111b464e99SMatthias Ringwald uint16_t lmp_subversion = little_endian_read_16(packet, 12); 1121b464e99SMatthias Ringwald printf("- HCI Version 0x%04x\n", hci_version); 1131b464e99SMatthias Ringwald printf("- HCI Revision 0x%04x\n", hci_revision); 1141b464e99SMatthias Ringwald printf("- LMP Version 0x%04x\n", lmp_version); 1151b464e99SMatthias Ringwald printf("- LMP Subversion 0x%04x\n", lmp_subversion); 1161b464e99SMatthias Ringwald printf("- Manufacturer 0x%04x\n", manufacturer); 1171b464e99SMatthias Ringwald } 1181b464e99SMatthias Ringwald 1191b464e99SMatthias Ringwald static void packet_handler (uint8_t packet_type, uint16_t channel, uint8_t *packet, uint16_t size){ 1201b464e99SMatthias Ringwald UNUSED(channel); 1211b464e99SMatthias Ringwald UNUSED(size); 1221b464e99SMatthias Ringwald if (packet_type != HCI_EVENT_PACKET) return; 1231b464e99SMatthias Ringwald switch (hci_event_packet_get_type(packet)) { 1241b464e99SMatthias Ringwald case BTSTACK_EVENT_STATE: 125*00f9da3aSMatthias Ringwald switch (btstack_event_state_get_state(packet)){ 126*00f9da3aSMatthias Ringwald case HCI_STATE_WORKING: 1271b464e99SMatthias Ringwald gap_local_bd_addr(local_addr); 1281b464e99SMatthias Ringwald if (using_static_address) { 1291b464e99SMatthias Ringwald memcpy(local_addr, static_address, 6); 1301b464e99SMatthias Ringwald } 1311b464e99SMatthias Ringwald printf("BTstack up and running on %s.\n", bd_addr_to_str(local_addr)); 1321b464e99SMatthias Ringwald strcpy(tlv_db_path, TLV_DB_PATH_PREFIX); 133208aaac2SMatthias Ringwald #ifndef Q_OS_WIN 134208aaac2SMatthias Ringwald // bd_addr_to_str use ":" which is not allowed in windows file names 1351b464e99SMatthias Ringwald strcat(tlv_db_path, bd_addr_to_str(local_addr)); 136208aaac2SMatthias Ringwald #endif 1371b464e99SMatthias Ringwald strcat(tlv_db_path, TLV_DB_PATH_POSTFIX); 1381b464e99SMatthias Ringwald tlv_impl = btstack_tlv_posix_init_instance(&tlv_context, tlv_db_path); 1391b464e99SMatthias Ringwald btstack_tlv_set_instance(tlv_impl, &tlv_context); 1401b464e99SMatthias Ringwald #ifdef ENABLE_CLASSIC 1411b464e99SMatthias Ringwald hci_set_link_key_db(btstack_link_key_db_tlv_get_instance(tlv_impl, &tlv_context)); 1421b464e99SMatthias Ringwald #endif 1431b464e99SMatthias Ringwald #ifdef ENABLE_BLE 1441b464e99SMatthias Ringwald le_device_db_tlv_configure(tlv_impl, &tlv_context); 1451b464e99SMatthias Ringwald #endif 1461b464e99SMatthias Ringwald break; 147*00f9da3aSMatthias Ringwald case HCI_STATE_OFF: 148*00f9da3aSMatthias Ringwald btstack_tlv_posix_deinit(&tlv_context); 149*00f9da3aSMatthias Ringwald if (!shutdown_triggered) break; 150*00f9da3aSMatthias Ringwald // reset stdin 151*00f9da3aSMatthias Ringwald btstack_stdin_reset(); 152*00f9da3aSMatthias Ringwald log_info("Good bye, see you.\n"); 153*00f9da3aSMatthias Ringwald exit(0); 154*00f9da3aSMatthias Ringwald break; 155*00f9da3aSMatthias Ringwald default: 156*00f9da3aSMatthias Ringwald break; 157*00f9da3aSMatthias Ringwald } 158*00f9da3aSMatthias Ringwald break; 1591b464e99SMatthias Ringwald case HCI_EVENT_COMMAND_COMPLETE: 1601b464e99SMatthias Ringwald if (HCI_EVENT_IS_COMMAND_COMPLETE(packet, hci_read_local_version_information)){ 1611b464e99SMatthias Ringwald local_version_information_handler(packet); 1621b464e99SMatthias Ringwald } 1631b464e99SMatthias Ringwald if (memcmp(packet, read_static_address_command_complete_prefix, sizeof(read_static_address_command_complete_prefix)) == 0){ 1641b464e99SMatthias Ringwald reverse_48(&packet[7], static_address); 1651b464e99SMatthias Ringwald gap_random_address_set(static_address); 1661b464e99SMatthias Ringwald using_static_address = 1; 1671b464e99SMatthias Ringwald } 1681b464e99SMatthias Ringwald break; 1691b464e99SMatthias Ringwald default: 1701b464e99SMatthias Ringwald break; 1711b464e99SMatthias Ringwald } 1721b464e99SMatthias Ringwald } 1731b464e99SMatthias Ringwald 174*00f9da3aSMatthias Ringwald static void trigger_shutdown(void){ 1751b464e99SMatthias Ringwald printf("CTRL-C - SIGINT received, shutting down..\n"); 1761b464e99SMatthias Ringwald log_info("sigint_handler: shutting down"); 177*00f9da3aSMatthias Ringwald shutdown_triggered = true; 1781b464e99SMatthias Ringwald hci_power_control(HCI_POWER_OFF); 1791b464e99SMatthias Ringwald } 1801b464e99SMatthias Ringwald 1811b464e99SMatthias Ringwald static int led_state = 0; 1821b464e99SMatthias Ringwald void hal_led_toggle(void){ 1831b464e99SMatthias Ringwald led_state = 1 - led_state; 1841b464e99SMatthias Ringwald printf("LED State %u\n", led_state); 1851b464e99SMatthias Ringwald } 1861b464e99SMatthias Ringwald 1871b464e99SMatthias Ringwald #define USB_MAX_PATH_LEN 7 1881b464e99SMatthias Ringwald int btstack_main(int argc, const char * argv[]); 1891b464e99SMatthias Ringwald 1901b464e99SMatthias Ringwald int main(int argc, char * argv[]){ 1911b464e99SMatthias Ringwald 192aaf1052fSMatthias Ringwald #ifndef __WIN32 1931b464e99SMatthias Ringwald uint8_t usb_path[USB_MAX_PATH_LEN]; 1941b464e99SMatthias Ringwald int usb_path_len = 0; 1951b464e99SMatthias Ringwald const char * usb_path_string = NULL; 1961b464e99SMatthias Ringwald if (argc >= 3 && strcmp(argv[1], "-u") == 0){ 1971b464e99SMatthias Ringwald // parse command line options for "-u 11:22:33" 1981b464e99SMatthias Ringwald usb_path_string = argv[2]; 1991b464e99SMatthias Ringwald printf("Specified USB Path: "); 2001b464e99SMatthias Ringwald while (1){ 2011b464e99SMatthias Ringwald char * delimiter; 2021b464e99SMatthias Ringwald int port = strtol(usb_path_string, &delimiter, 16); 2031b464e99SMatthias Ringwald usb_path[usb_path_len] = port; 2041b464e99SMatthias Ringwald usb_path_len++; 2051b464e99SMatthias Ringwald printf("%02x ", port); 2061b464e99SMatthias Ringwald if (!delimiter) break; 2071b464e99SMatthias Ringwald if (*delimiter != ':' && *delimiter != '-') break; 2081b464e99SMatthias Ringwald usb_path_string = delimiter+1; 2091b464e99SMatthias Ringwald } 2101b464e99SMatthias Ringwald printf("\n"); 2111b464e99SMatthias Ringwald argc -= 2; 2121b464e99SMatthias Ringwald memmove(&argv[1], &argv[3], (argc-1) * sizeof(char *)); 2131b464e99SMatthias Ringwald } 214aaf1052fSMatthias Ringwald #endif 2151b464e99SMatthias Ringwald 2161b464e99SMatthias Ringwald QCoreApplication a(argc, argv); 2171b464e99SMatthias Ringwald 2181b464e99SMatthias Ringwald /// GET STARTED with BTstack /// 2191b464e99SMatthias Ringwald btstack_memory_init(); 2201b464e99SMatthias Ringwald btstack_run_loop_init(btstack_run_loop_qt_get_instance()); 2211b464e99SMatthias Ringwald 222aaf1052fSMatthias Ringwald #ifndef __WIN32 2231b464e99SMatthias Ringwald if (usb_path_len){ 2241b464e99SMatthias Ringwald hci_transport_usb_set_path(usb_path_len, usb_path); 2251b464e99SMatthias Ringwald } 226aaf1052fSMatthias Ringwald #endif 2271b464e99SMatthias Ringwald 2287435ec7bSMatthias Ringwald // log into file using HCI_DUMP_PACKETLOGGER format 2291b464e99SMatthias Ringwald char pklg_path[100]; 230aaf1052fSMatthias Ringwald #ifdef __WIN32 231aaf1052fSMatthias Ringwald strcpy(pklg_path, "hci_dump"); 232aaf1052fSMatthias Ringwald #else 2331b464e99SMatthias Ringwald strcpy(pklg_path, "/tmp/hci_dump"); 2341b464e99SMatthias Ringwald if (usb_path_len){ 2351b464e99SMatthias Ringwald strcat(pklg_path, "_"); 2361b464e99SMatthias Ringwald strcat(pklg_path, usb_path_string); 2371b464e99SMatthias Ringwald } 238aaf1052fSMatthias Ringwald #endif 2391b464e99SMatthias Ringwald strcat(pklg_path, ".pklg"); 2407435ec7bSMatthias Ringwald hci_dump_posix_fs_open(pklg_path, HCI_DUMP_PACKETLOGGER); 2417435ec7bSMatthias Ringwald const hci_dump_t * hci_dump_impl = hci_dump_posix_fs_get_instance(); 2427435ec7bSMatthias Ringwald hci_dump_init(hci_dump_impl); 2431b464e99SMatthias Ringwald printf("Packet Log: %s\n", pklg_path); 2441b464e99SMatthias Ringwald 2451b464e99SMatthias Ringwald // init HCI 24684ede529SMatthias Ringwald #if 1 24784ede529SMatthias Ringwald // USB 2481b464e99SMatthias Ringwald hci_init(hci_transport_usb_instance(), NULL); 24984ede529SMatthias Ringwald #else 25084ede529SMatthias Ringwald // H4 25184ede529SMatthias Ringwald static hci_transport_config_uart_t config = { 25284ede529SMatthias Ringwald HCI_TRANSPORT_CONFIG_UART, 25384ede529SMatthias Ringwald 115200, 25484ede529SMatthias Ringwald 0, // main baudrate 25584ede529SMatthias Ringwald 1, // flow control 25684ede529SMatthias Ringwald NULL, 25784ede529SMatthias Ringwald }; 25884ede529SMatthias Ringwald config.device_name = "/dev/tty.usbserial-A900K2WS"; // DFROBOT 25984ede529SMatthias Ringwald 26084ede529SMatthias Ringwald // init HCI 261793a0509SMatthias Ringwald const btstack_uart_t * uart_driver = btstack_uart_block_instance(); 26284ede529SMatthias Ringwald const hci_transport_t * transport = hci_transport_h4_instance(uart_driver); 26384ede529SMatthias Ringwald hci_init(transport, (void*) &config); 26484ede529SMatthias Ringwald #endif 2651b464e99SMatthias Ringwald 2661b464e99SMatthias Ringwald #ifdef HAVE_PORTAUDIO 2671b464e99SMatthias Ringwald btstack_audio_sink_set_instance(btstack_audio_portaudio_sink_get_instance()); 2681b464e99SMatthias Ringwald btstack_audio_source_set_instance(btstack_audio_portaudio_source_get_instance()); 2691b464e99SMatthias Ringwald #endif 2701b464e99SMatthias Ringwald 2711b464e99SMatthias Ringwald // inform about BTstack state 2721b464e99SMatthias Ringwald hci_event_callback_registration.callback = &packet_handler; 2731b464e99SMatthias Ringwald hci_add_event_handler(&hci_event_callback_registration); 2741b464e99SMatthias Ringwald 275*00f9da3aSMatthias Ringwald // register callback for CTRL-c 276*00f9da3aSMatthias Ringwald #ifdef Q_OS_WIN 277*00f9da3aSMatthias Ringwald btstack_stdin_windows_init(); 278*00f9da3aSMatthias Ringwald btstack_stdin_window_register_ctrl_c_callback(&trigger_shutdown); 279*00f9da3aSMatthias Ringwald #else 280*00f9da3aSMatthias Ringwald btstack_signal_register_callback(SIGINT, &trigger_shutdown); 281*00f9da3aSMatthias Ringwald #endif 2821b464e99SMatthias Ringwald 2831b464e99SMatthias Ringwald // setup app 2841b464e99SMatthias Ringwald btstack_main(argc, (const char **) argv); 2851b464e99SMatthias Ringwald 2861b464e99SMatthias Ringwald // enter Qt run loop 2871b464e99SMatthias Ringwald return a.exec(); 2881b464e99SMatthias Ringwald } 289