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 23*2fca4dadSMilanka Ringwald * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL BLUEKITCHEN 24*2fca4dadSMilanka Ringwald * GMBH 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 // ***************************************************************************** 41ec8ae085SMilanka 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 1737bbeb3adSMilanka Ringwald if (packet_type != HCI_EVENT_PACKET) return; 1747bbeb3adSMilanka Ringwald 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 /* LISTING_END */ 21150514142SMatthias Ringwald 21250514142SMatthias Ringwald /* 21350514142SMatthias Ringwald * @section ATT Packet Handler 21450514142SMatthias Ringwald * 21550514142SMatthias Ringwald * @text The packet handler is used to setup and tear down the spp-over-gatt connection and its MTU 21650514142SMatthias Ringwald */ 21750514142SMatthias Ringwald 21850514142SMatthias Ringwald /* LISTING_START(packetHandler): Packet Handler */ 21950514142SMatthias Ringwald static void att_packet_handler (uint8_t packet_type, uint16_t channel, uint8_t *packet, uint16_t size){ 22050514142SMatthias Ringwald UNUSED(channel); 22150514142SMatthias Ringwald UNUSED(size); 22250514142SMatthias Ringwald 2237bbeb3adSMilanka Ringwald if (packet_type != HCI_EVENT_PACKET) return; 2247bbeb3adSMilanka Ringwald 22550514142SMatthias Ringwald int mtu; 22650514142SMatthias Ringwald nordic_spp_le_streamer_connection_t * context; 2277bbeb3adSMilanka Ringwald 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 /* LISTING_END */ 25750514142SMatthias Ringwald 25850514142SMatthias Ringwald 25950514142SMatthias Ringwald 260ebdf3c68SMilanka Ringwald /* 261ebdf3c68SMilanka Ringwald * @section Streamer 262ebdf3c68SMilanka Ringwald * 263ebdf3c68SMilanka Ringwald * @text The streamer function checks if notifications are enabled and if a notification can be sent now. 264ebdf3c68SMilanka Ringwald * It creates some test data - a single letter that gets increased every time - and tracks the data sent. 265ebdf3c68SMilanka Ringwald */ 266ebdf3c68SMilanka Ringwald 267ebdf3c68SMilanka Ringwald /* LISTING_START(streamer): Streaming code */ 268ebdf3c68SMilanka Ringwald static void nordic_can_send(void * some_context){ 269ebdf3c68SMilanka Ringwald UNUSED(some_context); 270ebdf3c68SMilanka Ringwald 271ebdf3c68SMilanka Ringwald // find next active streaming connection 272ebdf3c68SMilanka Ringwald int old_connection_index = connection_index; 273ebdf3c68SMilanka Ringwald while (1){ 274ebdf3c68SMilanka Ringwald // active found? 275ebdf3c68SMilanka Ringwald if ((nordic_spp_le_streamer_connections[connection_index].connection_handle != HCI_CON_HANDLE_INVALID) && 276ebdf3c68SMilanka Ringwald (nordic_spp_le_streamer_connections[connection_index].le_notification_enabled)) break; 277ebdf3c68SMilanka Ringwald 278ebdf3c68SMilanka Ringwald // check next 279ebdf3c68SMilanka Ringwald next_connection_index(); 280ebdf3c68SMilanka Ringwald 281ebdf3c68SMilanka Ringwald // none found 282ebdf3c68SMilanka Ringwald if (connection_index == old_connection_index) return; 283ebdf3c68SMilanka Ringwald } 284ebdf3c68SMilanka Ringwald 285ebdf3c68SMilanka Ringwald nordic_spp_le_streamer_connection_t * context = &nordic_spp_le_streamer_connections[connection_index]; 286ebdf3c68SMilanka Ringwald 287ebdf3c68SMilanka Ringwald // create test data 288ebdf3c68SMilanka Ringwald context->counter++; 289ebdf3c68SMilanka Ringwald if (context->counter > 'Z') context->counter = 'A'; 290ebdf3c68SMilanka Ringwald memset(context->test_data, context->counter, context->test_data_len); 291ebdf3c68SMilanka Ringwald 292ebdf3c68SMilanka Ringwald // send 293ebdf3c68SMilanka Ringwald nordic_spp_service_server_send(context->connection_handle, (uint8_t*) context->test_data, context->test_data_len); 294ebdf3c68SMilanka Ringwald 295ebdf3c68SMilanka Ringwald // track 296ebdf3c68SMilanka Ringwald test_track_sent(context, context->test_data_len); 297ebdf3c68SMilanka Ringwald 298ebdf3c68SMilanka Ringwald // request next send event 299ebdf3c68SMilanka Ringwald nordic_spp_service_server_request_can_send_now(&context->send_request, context->connection_handle); 300ebdf3c68SMilanka Ringwald 301ebdf3c68SMilanka Ringwald // check next 302ebdf3c68SMilanka Ringwald next_connection_index(); 303ebdf3c68SMilanka Ringwald } 304ebdf3c68SMilanka Ringwald /* LISTING_END */ 305ebdf3c68SMilanka Ringwald 306257f2b00SMatthias Ringwald static void nordic_spp_packet_handler(uint8_t packet_type, uint16_t channel, uint8_t *packet, uint16_t size){ 307257f2b00SMatthias Ringwald hci_con_handle_t con_handle; 308257f2b00SMatthias Ringwald nordic_spp_le_streamer_connection_t * context; 309257f2b00SMatthias Ringwald switch (packet_type){ 310257f2b00SMatthias Ringwald case HCI_EVENT_PACKET: 311257f2b00SMatthias Ringwald if (hci_event_packet_get_type(packet) != HCI_EVENT_GATTSERVICE_META) break; 312257f2b00SMatthias Ringwald switch (hci_event_gattservice_meta_get_subevent_code(packet)){ 313257f2b00SMatthias Ringwald case GATTSERVICE_SUBEVENT_SPP_SERVICE_CONNECTED: 314257f2b00SMatthias Ringwald con_handle = gattservice_subevent_spp_service_connected_get_con_handle(packet); 315257f2b00SMatthias Ringwald context = connection_for_conn_handle(con_handle); 316257f2b00SMatthias Ringwald if (!context) break; 317ebdf3c68SMilanka Ringwald context->le_notification_enabled = 1; 318ebdf3c68SMilanka Ringwald test_reset(context); 319ebdf3c68SMilanka Ringwald context->send_request.callback = &nordic_can_send; 320ebdf3c68SMilanka Ringwald nordic_spp_service_server_request_can_send_now(&context->send_request, context->connection_handle); 321257f2b00SMatthias Ringwald break; 322257f2b00SMatthias Ringwald case GATTSERVICE_SUBEVENT_SPP_SERVICE_DISCONNECTED: 323257f2b00SMatthias Ringwald con_handle = HCI_CON_HANDLE_INVALID; 324257f2b00SMatthias Ringwald context = connection_for_conn_handle(con_handle); 325257f2b00SMatthias Ringwald if (!context) break; 326257f2b00SMatthias Ringwald context->le_notification_enabled = 0; 327257f2b00SMatthias Ringwald break; 328257f2b00SMatthias Ringwald default: 329257f2b00SMatthias Ringwald break; 330257f2b00SMatthias Ringwald } 331257f2b00SMatthias Ringwald break; 332257f2b00SMatthias Ringwald case RFCOMM_DATA_PACKET: 333ebdf3c68SMilanka Ringwald printf("RECV: "); 334257f2b00SMatthias Ringwald printf_hexdump(packet, size); 335005e7bd2SMatthias Ringwald context = connection_for_conn_handle((hci_con_handle_t) channel); 336005e7bd2SMatthias Ringwald if (!context) break; 337ebdf3c68SMilanka Ringwald test_track_sent(context, size); 338257f2b00SMatthias Ringwald break; 339257f2b00SMatthias Ringwald default: 340257f2b00SMatthias Ringwald break; 341ebdf3c68SMilanka Ringwald } 342ebdf3c68SMilanka Ringwald } 343ebdf3c68SMilanka Ringwald 344ebdf3c68SMilanka Ringwald int btstack_main(void); 345ba7944beSMilanka Ringwald int btstack_main(void){ 346ebdf3c68SMilanka Ringwald // register for HCI events 34750514142SMatthias Ringwald hci_event_callback_registration.callback = &hci_packet_handler; 348ebdf3c68SMilanka Ringwald hci_add_event_handler(&hci_event_callback_registration); 349ebdf3c68SMilanka Ringwald 350ebdf3c68SMilanka Ringwald l2cap_init(); 351ebdf3c68SMilanka Ringwald 352ebdf3c68SMilanka Ringwald // setup LE device DB 353ebdf3c68SMilanka Ringwald le_device_db_init(); 354ebdf3c68SMilanka Ringwald 355ebdf3c68SMilanka Ringwald // setup SM: Display only 356ebdf3c68SMilanka Ringwald sm_init(); 357ebdf3c68SMilanka Ringwald 358ebdf3c68SMilanka Ringwald // setup ATT server 359ebdf3c68SMilanka Ringwald att_server_init(profile_data, NULL, NULL); 360ebdf3c68SMilanka Ringwald 361ebdf3c68SMilanka Ringwald // setup Nordic SPP service 362257f2b00SMatthias Ringwald nordic_spp_service_server_init(&nordic_spp_packet_handler); 363ebdf3c68SMilanka Ringwald 36450514142SMatthias Ringwald // register for ATT events 36550514142SMatthias Ringwald att_server_register_packet_handler(att_packet_handler); 36650514142SMatthias Ringwald 367ebdf3c68SMilanka Ringwald // setup advertisements 368ebdf3c68SMilanka Ringwald uint16_t adv_int_min = 0x0030; 369ebdf3c68SMilanka Ringwald uint16_t adv_int_max = 0x0030; 370ebdf3c68SMilanka Ringwald uint8_t adv_type = 0; 371ebdf3c68SMilanka Ringwald bd_addr_t null_addr; 372ebdf3c68SMilanka Ringwald memset(null_addr, 0, 6); 373ebdf3c68SMilanka Ringwald gap_advertisements_set_params(adv_int_min, adv_int_max, adv_type, 0, null_addr, 0x07, 0x00); 374ebdf3c68SMilanka Ringwald gap_advertisements_set_data(adv_data_len, (uint8_t*) adv_data); 375ebdf3c68SMilanka Ringwald gap_advertisements_enable(1); 376ebdf3c68SMilanka Ringwald 377ebdf3c68SMilanka Ringwald // init client state 378ebdf3c68SMilanka Ringwald init_connections(); 379ebdf3c68SMilanka Ringwald 380ebdf3c68SMilanka Ringwald // turn on! 381ebdf3c68SMilanka Ringwald hci_power_control(HCI_POWER_ON); 382ebdf3c68SMilanka Ringwald 383ebdf3c68SMilanka Ringwald return 0; 384ebdf3c68SMilanka Ringwald } 385ebdf3c68SMilanka Ringwald /* EXAMPLE_END */ 386