1ebdf3c68SMilanka Ringwald /* 2ebdf3c68SMilanka Ringwald * Copyright (C) 2014 BlueKitchen GmbH 3ebdf3c68SMilanka Ringwald * 4ebdf3c68SMilanka Ringwald * Redistribution and use in source and binary forms, with or without 5ebdf3c68SMilanka Ringwald * modification, are permitted provided that the following conditions 6ebdf3c68SMilanka Ringwald * are met: 7ebdf3c68SMilanka Ringwald * 8ebdf3c68SMilanka Ringwald * 1. Redistributions of source code must retain the above copyright 9ebdf3c68SMilanka Ringwald * notice, this list of conditions and the following disclaimer. 10ebdf3c68SMilanka Ringwald * 2. Redistributions in binary form must reproduce the above copyright 11ebdf3c68SMilanka Ringwald * notice, this list of conditions and the following disclaimer in the 12ebdf3c68SMilanka Ringwald * documentation and/or other materials provided with the distribution. 13ebdf3c68SMilanka Ringwald * 3. Neither the name of the copyright holders nor the names of 14ebdf3c68SMilanka Ringwald * contributors may be used to endorse or promote products derived 15ebdf3c68SMilanka Ringwald * from this software without specific prior written permission. 16ebdf3c68SMilanka Ringwald * 4. Any redistribution, use, or modification is done solely for 17ebdf3c68SMilanka Ringwald * personal benefit and not for any commercial purpose or for 18ebdf3c68SMilanka Ringwald * monetary gain. 19ebdf3c68SMilanka Ringwald * 20ebdf3c68SMilanka Ringwald * THIS SOFTWARE IS PROVIDED BY BLUEKITCHEN GMBH AND CONTRIBUTORS 21ebdf3c68SMilanka Ringwald * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT 22ebdf3c68SMilanka Ringwald * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS 23ebdf3c68SMilanka Ringwald * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL MATTHIAS 24ebdf3c68SMilanka Ringwald * RINGWALD OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, 25ebdf3c68SMilanka Ringwald * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, 26ebdf3c68SMilanka Ringwald * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS 27ebdf3c68SMilanka Ringwald * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED 28ebdf3c68SMilanka Ringwald * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, 29ebdf3c68SMilanka Ringwald * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF 30ebdf3c68SMilanka Ringwald * THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 31ebdf3c68SMilanka Ringwald * SUCH DAMAGE. 32ebdf3c68SMilanka Ringwald * 33ebdf3c68SMilanka Ringwald * Please inquire about commercial licensing options at 34ebdf3c68SMilanka Ringwald * [email protected] 35ebdf3c68SMilanka Ringwald * 36ebdf3c68SMilanka Ringwald */ 37ebdf3c68SMilanka Ringwald 38e501bae0SMatthias Ringwald #define BTSTACK_FILE__ "nordic_spp_le_streamer.c" 39ebdf3c68SMilanka Ringwald 40ebdf3c68SMilanka Ringwald // ***************************************************************************** 41*ec8ae085SMilanka Ringwald /* EXAMPLE_START(nordic_spp_le_streamer): LE Nordic SPP-like Streamer Server 42ebdf3c68SMilanka Ringwald * 43ebdf3c68SMilanka Ringwald * @text All newer operating systems provide GATT Client functionality. 44ebdf3c68SMilanka Ringwald * This example shows how to get a maximal throughput via BLE: 45ebdf3c68SMilanka Ringwald * - send whenever possible, 46ebdf3c68SMilanka Ringwald * - use the max ATT MTU. 47ebdf3c68SMilanka Ringwald * 48ebdf3c68SMilanka Ringwald * @text In theory, we should also update the connection parameters, but we already get 49ebdf3c68SMilanka Ringwald * a connection interval of 30 ms and there's no public way to use a shorter 50ebdf3c68SMilanka Ringwald * interval with iOS (if we're not implementing an HID device). 51ebdf3c68SMilanka Ringwald * 52ebdf3c68SMilanka Ringwald * @text Note: To start the streaming, run the example. 53ebdf3c68SMilanka Ringwald * On remote device use some GATT Explorer, e.g. LightBlue, BLExplr to enable notifications. 54ebdf3c68SMilanka Ringwald */ 55ebdf3c68SMilanka Ringwald // ***************************************************************************** 56ebdf3c68SMilanka Ringwald 57ebdf3c68SMilanka Ringwald #include <inttypes.h> 58ebdf3c68SMilanka Ringwald #include <stdint.h> 59ebdf3c68SMilanka Ringwald #include <stdio.h> 60ebdf3c68SMilanka Ringwald #include <stdlib.h> 61ebdf3c68SMilanka Ringwald #include <string.h> 62ebdf3c68SMilanka Ringwald 63ebdf3c68SMilanka Ringwald #include "btstack.h" 64ebdf3c68SMilanka Ringwald #include "ble/gatt-service/nordic_spp_service_server.h" 65ebdf3c68SMilanka Ringwald 66a63a688aSMatthias Ringwald // nordic_spp_le_streamer.gatt contains the declaration of the provided GATT Services + Characteristics 67a63a688aSMatthias Ringwald // nordic_spp_le_streamer.h contains the binary representation of nordic_spp_le_streamer.gatt 68a63a688aSMatthias Ringwald // it is generated by the build system by calling: $BTSTACK_ROOT/tool/compile_gatt.py nordic_spp_le_streamer.gatt nordic_spp_le_streamer.h 69a63a688aSMatthias Ringwald // it needs to be regenerated when the GATT Database declared in nordic_spp_le_streamer.gatt file is modified 70a63a688aSMatthias Ringwald #include "nordic_spp_le_streamer.h" 71a63a688aSMatthias Ringwald 72ebdf3c68SMilanka Ringwald #define REPORT_INTERVAL_MS 3000 73ebdf3c68SMilanka Ringwald #define MAX_NR_CONNECTIONS 3 74ebdf3c68SMilanka Ringwald 75ebdf3c68SMilanka Ringwald const uint8_t adv_data[] = { 76ebdf3c68SMilanka Ringwald // Flags general discoverable, BR/EDR not supported 77ebdf3c68SMilanka Ringwald 2, BLUETOOTH_DATA_TYPE_FLAGS, 0x06, 78ebdf3c68SMilanka Ringwald // Name 79ba7944beSMilanka Ringwald 8, BLUETOOTH_DATA_TYPE_COMPLETE_LOCAL_NAME, 'n', 'R', 'F',' ', 'S', 'P', 'P', 80ebdf3c68SMilanka Ringwald // UUID ... 814163d2ebSMilanka Ringwald 17, BLUETOOTH_DATA_TYPE_COMPLETE_LIST_OF_128_BIT_SERVICE_CLASS_UUIDS, 0x9e, 0xca, 0xdc, 0x24, 0xe, 0xe5, 0xa9, 0xe0, 0x93, 0xf3, 0xa3, 0xb5, 0x1, 0x0, 0x40, 0x6e, 82ebdf3c68SMilanka Ringwald }; 83ebdf3c68SMilanka Ringwald const uint8_t adv_data_len = sizeof(adv_data); 84ebdf3c68SMilanka Ringwald 85ebdf3c68SMilanka Ringwald static btstack_packet_callback_registration_t hci_event_callback_registration; 86ebdf3c68SMilanka Ringwald 87ebdf3c68SMilanka Ringwald // support for multiple clients 88ebdf3c68SMilanka Ringwald typedef struct { 89ebdf3c68SMilanka Ringwald char name; 90ebdf3c68SMilanka Ringwald int le_notification_enabled; 91ebdf3c68SMilanka Ringwald hci_con_handle_t connection_handle; 92ebdf3c68SMilanka Ringwald int counter; 93ebdf3c68SMilanka Ringwald char test_data[200]; 94ebdf3c68SMilanka Ringwald int test_data_len; 95ebdf3c68SMilanka Ringwald uint32_t test_data_sent; 96ebdf3c68SMilanka Ringwald uint32_t test_data_start; 97ebdf3c68SMilanka Ringwald btstack_context_callback_registration_t send_request; 98ebdf3c68SMilanka Ringwald } nordic_spp_le_streamer_connection_t; 99ebdf3c68SMilanka Ringwald 100ebdf3c68SMilanka Ringwald static nordic_spp_le_streamer_connection_t nordic_spp_le_streamer_connections[MAX_NR_CONNECTIONS]; 101ebdf3c68SMilanka Ringwald 102ebdf3c68SMilanka Ringwald // round robin sending 103ebdf3c68SMilanka Ringwald static int connection_index; 104ebdf3c68SMilanka Ringwald 105ebdf3c68SMilanka Ringwald static void init_connections(void){ 106ebdf3c68SMilanka Ringwald // track connections 107ebdf3c68SMilanka Ringwald int i; 108ebdf3c68SMilanka Ringwald for (i=0;i<MAX_NR_CONNECTIONS;i++){ 109ebdf3c68SMilanka Ringwald nordic_spp_le_streamer_connections[i].connection_handle = HCI_CON_HANDLE_INVALID; 110ebdf3c68SMilanka Ringwald nordic_spp_le_streamer_connections[i].name = 'A' + i; 111ebdf3c68SMilanka Ringwald } 112ebdf3c68SMilanka Ringwald } 113ebdf3c68SMilanka Ringwald 114ebdf3c68SMilanka Ringwald static nordic_spp_le_streamer_connection_t * connection_for_conn_handle(hci_con_handle_t conn_handle){ 115ebdf3c68SMilanka Ringwald int i; 116ebdf3c68SMilanka Ringwald for (i=0;i<MAX_NR_CONNECTIONS;i++){ 117ebdf3c68SMilanka Ringwald if (nordic_spp_le_streamer_connections[i].connection_handle == conn_handle) return &nordic_spp_le_streamer_connections[i]; 118ebdf3c68SMilanka Ringwald } 119ebdf3c68SMilanka Ringwald return NULL; 120ebdf3c68SMilanka Ringwald } 121ebdf3c68SMilanka Ringwald 122ebdf3c68SMilanka Ringwald static void next_connection_index(void){ 123ebdf3c68SMilanka Ringwald connection_index++; 124ebdf3c68SMilanka Ringwald if (connection_index == MAX_NR_CONNECTIONS){ 125ebdf3c68SMilanka Ringwald connection_index = 0; 126ebdf3c68SMilanka Ringwald } 127ebdf3c68SMilanka Ringwald } 128ebdf3c68SMilanka Ringwald 129ebdf3c68SMilanka Ringwald /* 130ebdf3c68SMilanka Ringwald * @section Track throughput 131ebdf3c68SMilanka Ringwald * @text We calculate the throughput by setting a start time and measuring the amount of 132ebdf3c68SMilanka Ringwald * data sent. After a configurable REPORT_INTERVAL_MS, we print the throughput in kB/s 133ebdf3c68SMilanka Ringwald * and reset the counter and start time. 134ebdf3c68SMilanka Ringwald */ 135ebdf3c68SMilanka Ringwald 136ebdf3c68SMilanka Ringwald /* LISTING_START(tracking): Tracking throughput */ 137ebdf3c68SMilanka Ringwald 138ebdf3c68SMilanka Ringwald static void test_reset(nordic_spp_le_streamer_connection_t * context){ 139ebdf3c68SMilanka Ringwald context->test_data_start = btstack_run_loop_get_time_ms(); 140ebdf3c68SMilanka Ringwald context->test_data_sent = 0; 141ebdf3c68SMilanka Ringwald } 142ebdf3c68SMilanka Ringwald 143ebdf3c68SMilanka Ringwald static void test_track_sent(nordic_spp_le_streamer_connection_t * context, int bytes_sent){ 144ebdf3c68SMilanka Ringwald context->test_data_sent += bytes_sent; 145ebdf3c68SMilanka Ringwald // evaluate 146ebdf3c68SMilanka Ringwald uint32_t now = btstack_run_loop_get_time_ms(); 147ebdf3c68SMilanka Ringwald uint32_t time_passed = now - context->test_data_start; 148ebdf3c68SMilanka Ringwald if (time_passed < REPORT_INTERVAL_MS) return; 149ebdf3c68SMilanka Ringwald // print speed 150ebdf3c68SMilanka Ringwald int bytes_per_second = context->test_data_sent * 1000 / time_passed; 151ebdf3c68SMilanka Ringwald printf("%c: %"PRIu32" bytes sent-> %u.%03u kB/s\n", context->name, context->test_data_sent, bytes_per_second / 1000, bytes_per_second % 1000); 152ebdf3c68SMilanka Ringwald 153ebdf3c68SMilanka Ringwald // restart 154ebdf3c68SMilanka Ringwald context->test_data_start = now; 155ebdf3c68SMilanka Ringwald context->test_data_sent = 0; 156ebdf3c68SMilanka Ringwald } 157ebdf3c68SMilanka Ringwald /* LISTING_END(tracking): Tracking throughput */ 158ebdf3c68SMilanka Ringwald 159ebdf3c68SMilanka Ringwald /* 16050514142SMatthias Ringwald * @section HCI Packet Handler 161ebdf3c68SMilanka Ringwald * 16250514142SMatthias Ringwald * @text The packet handler prints the welcome message and requests a connection paramter update for LE Connections 163ebdf3c68SMilanka Ringwald */ 164ebdf3c68SMilanka Ringwald 165ebdf3c68SMilanka Ringwald /* LISTING_START(packetHandler): Packet Handler */ 16650514142SMatthias Ringwald static void hci_packet_handler (uint8_t packet_type, uint16_t channel, uint8_t *packet, uint16_t size){ 167ebdf3c68SMilanka Ringwald UNUSED(channel); 168ebdf3c68SMilanka Ringwald UNUSED(size); 169ebdf3c68SMilanka Ringwald 170ebdf3c68SMilanka Ringwald uint16_t conn_interval; 17150514142SMatthias Ringwald hci_con_handle_t con_handle; 17250514142SMatthias Ringwald 173ebdf3c68SMilanka Ringwald switch (packet_type) { 174ebdf3c68SMilanka Ringwald case HCI_EVENT_PACKET: 175ebdf3c68SMilanka Ringwald switch (hci_event_packet_get_type(packet)) { 176ebdf3c68SMilanka Ringwald case BTSTACK_EVENT_STATE: 177ebdf3c68SMilanka Ringwald // BTstack activated, get started 178ebdf3c68SMilanka Ringwald if (btstack_event_state_get_state(packet) == HCI_STATE_WORKING) { 17950514142SMatthias Ringwald printf("To start the streaming, please run nRF Toolbox -> UART to connect.\n"); 180ebdf3c68SMilanka Ringwald } 181ebdf3c68SMilanka Ringwald break; 182ebdf3c68SMilanka Ringwald case HCI_EVENT_LE_META: 183ebdf3c68SMilanka Ringwald switch (hci_event_le_meta_get_subevent_code(packet)) { 184ebdf3c68SMilanka Ringwald case HCI_SUBEVENT_LE_CONNECTION_COMPLETE: 18550514142SMatthias Ringwald // print connection parameters (without using float operations) 18650514142SMatthias Ringwald con_handle = hci_subevent_le_connection_complete_get_connection_handle(packet); 18750514142SMatthias Ringwald conn_interval = hci_subevent_le_connection_complete_get_conn_interval(packet); 18850514142SMatthias Ringwald printf("LE Connection - Connection Interval: %u.%02u ms\n", conn_interval * 125 / 100, 25 * (conn_interval & 3)); 18950514142SMatthias Ringwald printf("LE Connection - Connection Latency: %u\n", hci_subevent_le_connection_complete_get_conn_latency(packet)); 19050514142SMatthias Ringwald 19150514142SMatthias Ringwald // request min con interval 15 ms for iOS 11+ 19250514142SMatthias Ringwald printf("LE Connection - Request 15 ms connection interval\n"); 19350514142SMatthias Ringwald gap_request_connection_parameter_update(con_handle, 12, 12, 0, 0x0048); 19450514142SMatthias Ringwald break; 19550514142SMatthias Ringwald case HCI_SUBEVENT_LE_CONNECTION_UPDATE_COMPLETE: 19650514142SMatthias Ringwald // print connection parameters (without using float operations) 19750514142SMatthias Ringwald con_handle = hci_subevent_le_connection_update_complete_get_connection_handle(packet); 19850514142SMatthias Ringwald conn_interval = hci_subevent_le_connection_update_complete_get_conn_interval(packet); 19950514142SMatthias Ringwald printf("LE Connection - Connection Param update - connection interval %u.%02u ms, latency %u\n", conn_interval * 125 / 100, 20050514142SMatthias Ringwald 25 * (conn_interval & 3), hci_subevent_le_connection_update_complete_get_conn_latency(packet)); 20150514142SMatthias Ringwald break; 20250514142SMatthias Ringwald default: 20350514142SMatthias Ringwald break; 20450514142SMatthias Ringwald } 20550514142SMatthias Ringwald break; 20650514142SMatthias Ringwald default: 20750514142SMatthias Ringwald break; 20850514142SMatthias Ringwald } 20950514142SMatthias Ringwald } 21050514142SMatthias Ringwald } 21150514142SMatthias Ringwald /* LISTING_END */ 21250514142SMatthias Ringwald 21350514142SMatthias Ringwald /* 21450514142SMatthias Ringwald * @section ATT Packet Handler 21550514142SMatthias Ringwald * 21650514142SMatthias Ringwald * @text The packet handler is used to setup and tear down the spp-over-gatt connection and its MTU 21750514142SMatthias Ringwald */ 21850514142SMatthias Ringwald 21950514142SMatthias Ringwald /* LISTING_START(packetHandler): Packet Handler */ 22050514142SMatthias Ringwald static void att_packet_handler (uint8_t packet_type, uint16_t channel, uint8_t *packet, uint16_t size){ 22150514142SMatthias Ringwald UNUSED(channel); 22250514142SMatthias Ringwald UNUSED(size); 22350514142SMatthias Ringwald 22450514142SMatthias Ringwald int mtu; 22550514142SMatthias Ringwald nordic_spp_le_streamer_connection_t * context; 22650514142SMatthias Ringwald switch (packet_type) { 22750514142SMatthias Ringwald case HCI_EVENT_PACKET: 22850514142SMatthias Ringwald switch (hci_event_packet_get_type(packet)) { 22950514142SMatthias Ringwald case ATT_EVENT_CONNECTED: 230ebdf3c68SMilanka Ringwald // setup new 231ebdf3c68SMilanka Ringwald context = connection_for_conn_handle(HCI_CON_HANDLE_INVALID); 232ebdf3c68SMilanka Ringwald if (!context) break; 233ebdf3c68SMilanka Ringwald context->counter = 'A'; 234ebdf3c68SMilanka Ringwald context->test_data_len = ATT_DEFAULT_MTU - 4; // -1 for nordic 0x01 packet type 23550514142SMatthias Ringwald context->connection_handle = att_event_connected_get_handle(packet); 236ebdf3c68SMilanka Ringwald break; 237ebdf3c68SMilanka Ringwald case ATT_EVENT_MTU_EXCHANGE_COMPLETE: 238ebdf3c68SMilanka Ringwald mtu = att_event_mtu_exchange_complete_get_MTU(packet) - 3; 239ebdf3c68SMilanka Ringwald context = connection_for_conn_handle(att_event_mtu_exchange_complete_get_handle(packet)); 240ebdf3c68SMilanka Ringwald if (!context) break; 241ebdf3c68SMilanka Ringwald context->test_data_len = btstack_min(mtu - 3, sizeof(context->test_data)); 242ebdf3c68SMilanka Ringwald printf("%c: ATT MTU = %u => use test data of len %u\n", context->name, mtu, context->test_data_len); 243ebdf3c68SMilanka Ringwald break; 24450514142SMatthias Ringwald case ATT_EVENT_DISCONNECTED: 24550514142SMatthias Ringwald context = connection_for_conn_handle(att_event_disconnected_get_handle(packet)); 24650514142SMatthias Ringwald if (!context) break; 24750514142SMatthias Ringwald // free connection 24850514142SMatthias Ringwald printf("%c: Disconnect\n", context->name); 24950514142SMatthias Ringwald context->le_notification_enabled = 0; 25050514142SMatthias Ringwald context->connection_handle = HCI_CON_HANDLE_INVALID; 25150514142SMatthias Ringwald break; 252ebdf3c68SMilanka Ringwald default: 253ebdf3c68SMilanka Ringwald break; 254ebdf3c68SMilanka Ringwald } 255ebdf3c68SMilanka Ringwald } 256ebdf3c68SMilanka Ringwald } 257ebdf3c68SMilanka Ringwald /* LISTING_END */ 25850514142SMatthias Ringwald 25950514142SMatthias Ringwald 26050514142SMatthias Ringwald 261ebdf3c68SMilanka Ringwald /* 262ebdf3c68SMilanka Ringwald * @section Streamer 263ebdf3c68SMilanka Ringwald * 264ebdf3c68SMilanka Ringwald * @text The streamer function checks if notifications are enabled and if a notification can be sent now. 265ebdf3c68SMilanka Ringwald * It creates some test data - a single letter that gets increased every time - and tracks the data sent. 266ebdf3c68SMilanka Ringwald */ 267ebdf3c68SMilanka Ringwald 268ebdf3c68SMilanka Ringwald /* LISTING_START(streamer): Streaming code */ 269ebdf3c68SMilanka Ringwald static void nordic_can_send(void * some_context){ 270ebdf3c68SMilanka Ringwald UNUSED(some_context); 271ebdf3c68SMilanka Ringwald 272ebdf3c68SMilanka Ringwald // find next active streaming connection 273ebdf3c68SMilanka Ringwald int old_connection_index = connection_index; 274ebdf3c68SMilanka Ringwald while (1){ 275ebdf3c68SMilanka Ringwald // active found? 276ebdf3c68SMilanka Ringwald if ((nordic_spp_le_streamer_connections[connection_index].connection_handle != HCI_CON_HANDLE_INVALID) && 277ebdf3c68SMilanka Ringwald (nordic_spp_le_streamer_connections[connection_index].le_notification_enabled)) break; 278ebdf3c68SMilanka Ringwald 279ebdf3c68SMilanka Ringwald // check next 280ebdf3c68SMilanka Ringwald next_connection_index(); 281ebdf3c68SMilanka Ringwald 282ebdf3c68SMilanka Ringwald // none found 283ebdf3c68SMilanka Ringwald if (connection_index == old_connection_index) return; 284ebdf3c68SMilanka Ringwald } 285ebdf3c68SMilanka Ringwald 286ebdf3c68SMilanka Ringwald nordic_spp_le_streamer_connection_t * context = &nordic_spp_le_streamer_connections[connection_index]; 287ebdf3c68SMilanka Ringwald 288ebdf3c68SMilanka Ringwald // create test data 289ebdf3c68SMilanka Ringwald context->counter++; 290ebdf3c68SMilanka Ringwald if (context->counter > 'Z') context->counter = 'A'; 291ebdf3c68SMilanka Ringwald memset(context->test_data, context->counter, context->test_data_len); 292ebdf3c68SMilanka Ringwald 293ebdf3c68SMilanka Ringwald // send 294ebdf3c68SMilanka Ringwald nordic_spp_service_server_send(context->connection_handle, (uint8_t*) context->test_data, context->test_data_len); 295ebdf3c68SMilanka Ringwald 296ebdf3c68SMilanka Ringwald // track 297ebdf3c68SMilanka Ringwald test_track_sent(context, context->test_data_len); 298ebdf3c68SMilanka Ringwald 299ebdf3c68SMilanka Ringwald // request next send event 300ebdf3c68SMilanka Ringwald nordic_spp_service_server_request_can_send_now(&context->send_request, context->connection_handle); 301ebdf3c68SMilanka Ringwald 302ebdf3c68SMilanka Ringwald // check next 303ebdf3c68SMilanka Ringwald next_connection_index(); 304ebdf3c68SMilanka Ringwald } 305ebdf3c68SMilanka Ringwald /* LISTING_END */ 306ebdf3c68SMilanka Ringwald 307ebdf3c68SMilanka Ringwald static void nordic_data_received(hci_con_handle_t tx_con_handle, const uint8_t * data, uint16_t size){ 308ebdf3c68SMilanka Ringwald nordic_spp_le_streamer_connection_t * context = connection_for_conn_handle(tx_con_handle); 309ebdf3c68SMilanka Ringwald 310ebdf3c68SMilanka Ringwald if (!context) return; 311ebdf3c68SMilanka Ringwald 312ebdf3c68SMilanka Ringwald if (size == 0 && context->le_notification_enabled == 0){ 313ebdf3c68SMilanka Ringwald context->le_notification_enabled = 1; 314ebdf3c68SMilanka Ringwald test_reset(context); 315ebdf3c68SMilanka Ringwald context->send_request.callback = &nordic_can_send; 316ebdf3c68SMilanka Ringwald nordic_spp_service_server_request_can_send_now(&context->send_request, context->connection_handle); 317ebdf3c68SMilanka Ringwald } else { 318ebdf3c68SMilanka Ringwald printf("RECV: "); 319ebdf3c68SMilanka Ringwald printf_hexdump(data, size); 320ebdf3c68SMilanka Ringwald test_track_sent(context, size); 321ebdf3c68SMilanka Ringwald } 322ebdf3c68SMilanka Ringwald } 323ebdf3c68SMilanka Ringwald 324ebdf3c68SMilanka Ringwald int btstack_main(void); 325ba7944beSMilanka Ringwald int btstack_main(void){ 326ebdf3c68SMilanka Ringwald // register for HCI events 32750514142SMatthias Ringwald hci_event_callback_registration.callback = &hci_packet_handler; 328ebdf3c68SMilanka Ringwald hci_add_event_handler(&hci_event_callback_registration); 329ebdf3c68SMilanka Ringwald 330ebdf3c68SMilanka Ringwald l2cap_init(); 331ebdf3c68SMilanka Ringwald 332ebdf3c68SMilanka Ringwald // setup LE device DB 333ebdf3c68SMilanka Ringwald le_device_db_init(); 334ebdf3c68SMilanka Ringwald 335ebdf3c68SMilanka Ringwald // setup SM: Display only 336ebdf3c68SMilanka Ringwald sm_init(); 337ebdf3c68SMilanka Ringwald 338ebdf3c68SMilanka Ringwald // setup ATT server 339ebdf3c68SMilanka Ringwald att_server_init(profile_data, NULL, NULL); 340ebdf3c68SMilanka Ringwald 341ebdf3c68SMilanka Ringwald // setup Nordic SPP service 342ebdf3c68SMilanka Ringwald nordic_spp_service_server_init(&nordic_data_received); 343ebdf3c68SMilanka Ringwald 34450514142SMatthias Ringwald // register for ATT events 34550514142SMatthias Ringwald att_server_register_packet_handler(att_packet_handler); 34650514142SMatthias Ringwald 347ebdf3c68SMilanka Ringwald // setup advertisements 348ebdf3c68SMilanka Ringwald uint16_t adv_int_min = 0x0030; 349ebdf3c68SMilanka Ringwald uint16_t adv_int_max = 0x0030; 350ebdf3c68SMilanka Ringwald uint8_t adv_type = 0; 351ebdf3c68SMilanka Ringwald bd_addr_t null_addr; 352ebdf3c68SMilanka Ringwald memset(null_addr, 0, 6); 353ebdf3c68SMilanka Ringwald gap_advertisements_set_params(adv_int_min, adv_int_max, adv_type, 0, null_addr, 0x07, 0x00); 354ebdf3c68SMilanka Ringwald gap_advertisements_set_data(adv_data_len, (uint8_t*) adv_data); 355ebdf3c68SMilanka Ringwald gap_advertisements_enable(1); 356ebdf3c68SMilanka Ringwald 357ebdf3c68SMilanka Ringwald // init client state 358ebdf3c68SMilanka Ringwald init_connections(); 359ebdf3c68SMilanka Ringwald 360ebdf3c68SMilanka Ringwald // turn on! 361ebdf3c68SMilanka Ringwald hci_power_control(HCI_POWER_ON); 362ebdf3c68SMilanka Ringwald 363ebdf3c68SMilanka Ringwald return 0; 364ebdf3c68SMilanka Ringwald } 365ebdf3c68SMilanka Ringwald /* EXAMPLE_END */ 366