/* * Copyright (C) 2011-2013 by Matthias Ringwald * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * 3. Neither the name of the copyright holders nor the names of * contributors may be used to endorse or promote products derived * from this software without specific prior written permission. * 4. This software may not be used in a commercial product * without an explicit license granted by the copyright holder. * * THIS SOFTWARE IS PROVIDED BY MATTHIAS RINGWALD AND CONTRIBUTORS * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL MATTHIAS * RINGWALD OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF * THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF * SUCH DAMAGE. * */ //***************************************************************************** // // BLE Peripheral Demo // //***************************************************************************** #include #include #include #include #include #include #include #include "btstack-config.h" #include #include "debug.h" #include "btstack_memory.h" #include "hci.h" #include "hci_dump.h" #include "l2cap.h" #include "sm.h" #include "att.h" #include "att_server.h" #include "gap_le.h" #include "central_device_db.h" #define HEARTBEAT_PERIOD_MS 1000 // test profile #include "profile.h" typedef enum { DISABLE_ADVERTISEMENTS = 1 << 0, SET_ADVERTISEMENT_PARAMS = 1 << 1, SET_ADVERTISEMENT_DATA = 1 << 2, SET_SCAN_RESPONSE_DATA = 1 << 3, ENABLE_ADVERTISEMENTS = 1 << 4, } todo_t; static todo_t todos = 0; ///------ static int advertisements_enabled = 0; static int gap_discoverable = 1; static int gap_connectable = 1; static int gap_bondable = 1; static int gap_directed_connectable = 0; static int gap_privacy = 1; static timer_source_t heartbeat; static uint8_t counter = 0; static int update_client = 0; static int client_configuration = 0; static uint16_t handle = 0; static void app_run(); static void show_usage(); static void update_advertisements(); // -> hier Adresse vom Tester eintragen für Directed Connectable Mode static bd_addr_t tester_address = {0x00, 0x1B, 0xDC, 0x06, 0x07, 0x5F}; static int tester_address_type = 0; // some test data static uint8_t adv_data_0[] = { 2, 01, 05, 03, 02, 0xf0, 0xff }; // AD Manufacturer Specific Data - Ericsson, 1, 2, 3, 4 static uint8_t adv_data_1[] = { 7, 0xff, 0x00, 0x00, 1, 2, 3, 4 }; // AD Local Name - 'BTstack' static uint8_t adv_data_2[] = { 8, 0x09, 'B', 'T', 's', 't', 'a', 'c', 'k' }; // AD Flags - 2 - General Discoverable mode static uint8_t adv_data_3[] = { 2, 01, 02 }; // AD Service Data - 0x1812 HID over LE static uint8_t adv_data_4[] = { 3, 0x16, 0x12, 0x18 }; // AD Service Solicitation - 0x1812 HID over LE static uint8_t adv_data_5[] = { 3, 0x14, 0x12, 0x18 }; // AD Services static uint8_t adv_data_6[] = { 3, 0x03, 0x12, 0x18 }; // AD Slave Preferred Connection Interval Range - no min, no max static uint8_t adv_data_7[] = { 5, 0x12, 0xff, 0xff, 0xff, 0xff }; // AD Tx Power Level - +4 dBm static uint8_t adv_data_8[] = { 2, 0x0a, 4 }; static uint8_t adv_data_len; static uint8_t adv_data[32]; typedef struct { uint16_t len; uint8_t * data; } advertisement_t; #define ADV(a) { sizeof(a), &a[0]} static advertisement_t advertisements[] = { ADV(adv_data_0), ADV(adv_data_1), ADV(adv_data_2), ADV(adv_data_3), ADV(adv_data_4), ADV(adv_data_5), ADV(adv_data_6), ADV(adv_data_7), ADV(adv_data_8), }; static int advertisement_index = 0; static void heartbeat_handler(struct timer *ts){ // restart timer run_loop_set_timer(ts, HEARTBEAT_PERIOD_MS); run_loop_add_timer(ts); counter++; update_client = 1; app_run(); } static void app_run(){ if (!update_client) return; if (!att_server_can_send()) return; int result = -1; switch (client_configuration){ case 0x01: printf("Notify value %u\n", counter); result = att_server_notify(0x0f, &counter, 1); break; case 0x02: printf("Indicate value %u\n", counter); result = att_server_indicate(0x0f, &counter, 1); break; default: return; } if (result){ printf("Error 0x%02x\n", result); return; } update_client = 0; } // write requests static int att_write_callback(uint16_t handle, uint16_t transaction_mode, uint16_t offset, uint8_t *buffer, uint16_t buffer_size, signature_t * signature){ printf("WRITE Callback, handle %04x\n", handle); switch(handle){ case 0x0010: client_configuration = buffer[0]; printf("Client Configuration set to %u\n", client_configuration); break; default: printf("Value: "); hexdump(buffer, buffer_size); break; } return 1; } static uint8_t gap_adv_type(){ if (gap_connectable){ if (gap_directed_connectable){ return 0x01; } return 0x00; } return 0x03; } static void gap_run(){ if (!hci_can_send_packet_now(HCI_COMMAND_DATA_PACKET)) return; if (todos & DISABLE_ADVERTISEMENTS){ todos &= ~DISABLE_ADVERTISEMENTS; advertisements_enabled = 0; hci_send_cmd(&hci_le_set_advertise_enable, 0); return; } if (todos & SET_ADVERTISEMENT_DATA){ todos &= ~SET_ADVERTISEMENT_DATA; hci_send_cmd(&hci_le_set_advertising_data, adv_data_len, adv_data); return; } // if (todos & SET_ADVERTISEMENT_PARAMS){ // todos &= ~SET_ADVERTISEMENT_PARAMS; // hci_send_cmd(&hci_le_set_advertising_parameters, 0x0800, 0x0800, gap_adv_type(), gap_privacy, tester_address_type, &tester_address, 0x07, 0x00); // return; // } if (todos & SET_SCAN_RESPONSE_DATA){ todos &= ~SET_SCAN_RESPONSE_DATA; hci_send_cmd(&hci_le_set_scan_response_data, adv_data_len, adv_data); return; } if (todos & ENABLE_ADVERTISEMENTS){ todos &= ~ENABLE_ADVERTISEMENTS; advertisements_enabled = 1; hci_send_cmd(&hci_le_set_advertise_enable, 1); show_usage(); return; } } static void app_packet_handler (uint8_t packet_type, uint16_t channel, uint8_t *packet, uint16_t size){ switch (packet_type) { case HCI_EVENT_PACKET: switch (packet[0]) { case BTSTACK_EVENT_STATE: // bt stack activated, get started if (packet[2] == HCI_STATE_WORKING) { printf("SM Init completed\n"); todos = SET_ADVERTISEMENT_PARAMS | SET_ADVERTISEMENT_DATA | SET_SCAN_RESPONSE_DATA | ENABLE_ADVERTISEMENTS; update_advertisements(); gap_run(); } break; case HCI_EVENT_LE_META: switch (packet[2]) { case HCI_SUBEVENT_LE_CONNECTION_COMPLETE: advertisements_enabled = 0; handle = READ_BT_16(packet, 4); printf("Connection handle 0x%04x", handle); // request connection parameter update - test parameters // l2cap_le_request_connection_parameter_update(READ_BT_16(packet, 4), 20, 1000, 100, 100); break; default: break; } break; case HCI_EVENT_DISCONNECTION_COMPLETE: if (!advertisements_enabled == 0 && gap_discoverable){ todos = ENABLE_ADVERTISEMENTS; } break; case SM_PASSKEY_DISPLAY_NUMBER: { // display number sm_event_t * event = (sm_event_t *) packet; printf("GAP Bonding: Display Passkey '%06u\n", event->passkey); break; } case SM_PASSKEY_DISPLAY_CANCEL: printf("GAP Bonding: Display cancel\n"); break; case SM_AUTHORIZATION_REQUEST: { // auto-authorize connection if requested sm_event_t * event = (sm_event_t *) packet; sm_authorization_grant(event->addr_type, event->address); break; } case ATT_HANDLE_VALUE_INDICATION_COMPLETE: printf("ATT_HANDLE_VALUE_INDICATION_COMPLETE status %u\n", packet[2]); break; default: break; } } gap_run(); } void setup(void){ /// GET STARTED with BTstack /// btstack_memory_init(); run_loop_init(RUN_LOOP_POSIX); // use logger: format HCI_DUMP_PACKETLOGGER, HCI_DUMP_BLUEZ or HCI_DUMP_STDOUT hci_dump_open("/tmp/hci_dump.pklg", HCI_DUMP_PACKETLOGGER); // init HCI hci_transport_t * transport = hci_transport_usb_instance(); hci_uart_config_t * config = NULL; bt_control_t * control = NULL; remote_device_db_t * remote_db = (remote_device_db_t *) &remote_device_db_memory; hci_init(transport, config, control, remote_db); // set up l2cap_le l2cap_init(); // setup central device db central_device_db_init(); // setup SM: Display only sm_init(); sm_set_io_capabilities(IO_CAPABILITY_DISPLAY_ONLY); sm_set_authentication_requirements( SM_AUTHREQ_BONDING | SM_AUTHREQ_MITM_PROTECTION); // sm_set_request_security(1); // sm_set_encryption_key_size_range(7,15); // setup ATT server att_server_init(profile_data, NULL, att_write_callback); att_server_register_packet_handler(app_packet_handler); } void show_usage(){ printf("\n--- CLI for LE Peripheral ---\n"); printf("Status: discoverable %u, connectable %u, bondable %u, directed connectable %u, privacy %u, ads enabled %u \n", gap_discoverable, gap_connectable, gap_bondable, gap_directed_connectable, gap_privacy, advertisements_enabled); printf("---\n"); printf("b - bondable off\n"); printf("B - bondable on\n"); printf("c - connectable off\n"); printf("C - connectable on\n"); printf("d - discoverable off\n"); printf("D - discoverable on\n"); printf("p - privacy off\n"); printf("P - privacy on\n"); printf("x - directed connectable off\n"); printf("X - directed connectable on\n"); printf("---\n"); printf("1 - AD Manufacturer Specific Data\n"); printf("2 - AD Local Name\n"); printf("3 - AD Flags\n"); printf("4 - AD Service Data\n"); printf("5 - AD Service Solicitation\n"); printf("6 - AD Services\n"); printf("7 - AD Slave Preferred Connection Interval Range\n"); printf("8 - AD Tx Power Level\n"); printf("---\n"); printf("t - terminate connection\n"); printf("---\n"); printf("Ctrl-c - exit\n"); printf("---\n"); } void update_advertisements(){ memset(adv_data, 0, 32); memcpy(adv_data, advertisements[advertisement_index].data, advertisements[advertisement_index].len); if (!gap_discoverable){ gap_connectable = 0; } if (!gap_connectable){ gap_bondable = 0; } if (!advertisements_enabled) return; // update all todos = DISABLE_ADVERTISEMENTS | SET_ADVERTISEMENT_PARAMS | SET_ADVERTISEMENT_DATA | SET_SCAN_RESPONSE_DATA | ENABLE_ADVERTISEMENTS; gap_run(); } int stdin_process(struct data_source *ds){ char buffer; read(ds->fd, &buffer, 1); switch (buffer){ case 'b': gap_bondable = 0; sm_set_authentication_requirements(SM_AUTHREQ_NO_BONDING); show_usage(); break; case 'B': gap_bondable = 1; sm_set_authentication_requirements(SM_AUTHREQ_BONDING); show_usage(); break; case 'c': gap_connectable = 0; update_advertisements(); break; case 'C': gap_connectable = 1; update_advertisements(); break; case 'd': gap_discoverable = 0; update_advertisements(); break; case 'D': gap_discoverable = 1; update_advertisements(); break; case 'p': gap_privacy = 0; update_advertisements(); break; case 'P': gap_privacy = 1; update_advertisements(); break; case 'x': gap_directed_connectable = 0; update_advertisements(); break; case 'X': gap_directed_connectable = 1; update_advertisements(); break; case '1': case '2': case '3': case '4': case '5': case '6': case '7': case '8': advertisement_index = buffer - '0'; update_advertisements(); break; case 't': hci_send_cmd(&hci_disconnect, handle, 0x13); break; default: show_usage(); break; } return 0; } static data_source_t stdin_source; void setup_cli(){ struct termios term = {0}; if (tcgetattr(0, &term) < 0) perror("tcsetattr()"); term.c_lflag &= ~ICANON; term.c_lflag &= ~ECHO; term.c_cc[VMIN] = 1; term.c_cc[VTIME] = 0; if (tcsetattr(0, TCSANOW, &term) < 0) perror("tcsetattr ICANON"); stdin_source.fd = 0; // stdin stdin_source.process = &stdin_process; run_loop_add_data_source(&stdin_source); } int main(void) { printf("BTstack LE Peripheral starting up...\n"); setup(); setup_cli(); gap_random_address_set_update_period(60000); gap_random_address_set_mode(GAP_RANDOM_ADDRESS_RESOLVABLE); // set one-shot timer heartbeat.process = &heartbeat_handler; run_loop_set_timer(&heartbeat, HEARTBEAT_PERIOD_MS); run_loop_add_timer(&heartbeat); // turn on! hci_power_control(HCI_POWER_ON); // go! run_loop_execute(); // happy compiler! return 0; }