1ffb44729SMilanka Ringwald /* 2ffb44729SMilanka Ringwald * Copyright (C) 2018 BlueKitchen GmbH 3ffb44729SMilanka Ringwald * 4ffb44729SMilanka Ringwald * Redistribution and use in source and binary forms, with or without 5ffb44729SMilanka Ringwald * modification, are permitted provided that the following conditions 6ffb44729SMilanka Ringwald * are met: 7ffb44729SMilanka Ringwald * 8ffb44729SMilanka Ringwald * 1. Redistributions of source code must retain the above copyright 9ffb44729SMilanka Ringwald * notice, this list of conditions and the following disclaimer. 10ffb44729SMilanka Ringwald * 2. Redistributions in binary form must reproduce the above copyright 11ffb44729SMilanka Ringwald * notice, this list of conditions and the following disclaimer in the 12ffb44729SMilanka Ringwald * documentation and/or other materials provided with the distribution. 13ffb44729SMilanka Ringwald * 3. Neither the name of the copyright holders nor the names of 14ffb44729SMilanka Ringwald * contributors may be used to endorse or promote products derived 15ffb44729SMilanka Ringwald * from this software without specific prior written permission. 16ffb44729SMilanka Ringwald * 4. Any redistribution, use, or modification is done solely for 17ffb44729SMilanka Ringwald * personal benefit and not for any commercial purpose or for 18ffb44729SMilanka Ringwald * monetary gain. 19ffb44729SMilanka Ringwald * 20ffb44729SMilanka Ringwald * THIS SOFTWARE IS PROVIDED BY BLUEKITCHEN GMBH AND CONTRIBUTORS 21ffb44729SMilanka Ringwald * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT 22ffb44729SMilanka 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, 25ffb44729SMilanka Ringwald * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, 26ffb44729SMilanka Ringwald * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS 27ffb44729SMilanka Ringwald * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED 28ffb44729SMilanka Ringwald * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, 29ffb44729SMilanka Ringwald * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF 30ffb44729SMilanka Ringwald * THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 31ffb44729SMilanka Ringwald * SUCH DAMAGE. 32ffb44729SMilanka Ringwald * 33ffb44729SMilanka Ringwald * Please inquire about commercial licensing options at 34ffb44729SMilanka Ringwald * [email protected] 35ffb44729SMilanka Ringwald * 36ffb44729SMilanka Ringwald */ 37ffb44729SMilanka Ringwald 38ffb44729SMilanka Ringwald #define BTSTACK_FILE__ "scan_parameters_service_server.c" 39ffb44729SMilanka Ringwald 40ffb44729SMilanka Ringwald /** 41ffb44729SMilanka Ringwald * Implementation of the Nordic SPP-like profile 42ffb44729SMilanka Ringwald * 43ffb44729SMilanka Ringwald * To use with your application, add '#import <scan_parameters_service.gatt' to your .gatt file 44ffb44729SMilanka Ringwald * and call all functions below. All strings and blobs need to stay valid after calling the functions. 45ffb44729SMilanka Ringwald */ 46ffb44729SMilanka Ringwald 47ffb44729SMilanka Ringwald #include <stdint.h> 48ffb44729SMilanka Ringwald #include <string.h> 49ffb44729SMilanka Ringwald 5082879dc3SMilanka Ringwald #ifdef ENABLE_TESTING_SUPPORT 5182879dc3SMilanka Ringwald #include <stdio.h> 5282879dc3SMilanka Ringwald #endif 5382879dc3SMilanka Ringwald 54ffb44729SMilanka Ringwald #include "btstack_defines.h" 55ffb44729SMilanka Ringwald #include "ble/att_db.h" 56ffb44729SMilanka Ringwald #include "ble/att_server.h" 57ffb44729SMilanka Ringwald #include "btstack_util.h" 58ffb44729SMilanka Ringwald #include "bluetooth_gatt.h" 59ffb44729SMilanka Ringwald #include "btstack_debug.h" 60ffb44729SMilanka Ringwald 61ffb44729SMilanka Ringwald #include "ble/gatt-service/scan_parameters_service_server.h" 62ffb44729SMilanka Ringwald 63ffb44729SMilanka Ringwald static btstack_packet_handler_t scan_parameters_packet_handler; 64ffb44729SMilanka Ringwald static btstack_context_callback_registration_t scan_parameters_callback; 65ffb44729SMilanka Ringwald static att_service_handler_t scan_parameters_service; 66ffb44729SMilanka Ringwald 67ffb44729SMilanka Ringwald static uint16_t scan_interval_window_value_handle; 68ffb44729SMilanka Ringwald static uint16_t scan_interval_window_value_handle_client_configuration; 69ffb44729SMilanka Ringwald 70ffb44729SMilanka Ringwald 71ffb44729SMilanka Ringwald static uint16_t scan_refresh_value_client_configuration; 72ffb44729SMilanka Ringwald static hci_con_handle_t scan_refresh_value_client_configuration_connection; 73ffb44729SMilanka Ringwald 74ffb44729SMilanka Ringwald static uint16_t scan_refresh_value_handle; 75ffb44729SMilanka Ringwald static uint16_t scan_refresh_value_handle_client_configuration; 76ffb44729SMilanka Ringwald 7789974af9SMilanka Ringwald static void scan_parameters_service_emit_state(hci_con_handle_t con_handle, uint16_t max_scan_interval, uint16_t min_scan_window){ 7889974af9SMilanka Ringwald if (scan_parameters_packet_handler == NULL) return; 7989974af9SMilanka Ringwald 8089974af9SMilanka Ringwald uint8_t event[9]; 8189974af9SMilanka Ringwald uint16_t pos = 0; 8289974af9SMilanka Ringwald event[pos++] = HCI_EVENT_GATTSERVICE_META; 8389974af9SMilanka Ringwald event[pos++] = sizeof(event) - 2; 8489974af9SMilanka Ringwald event[pos++] = GATTSERVICE_SUBEVENT_SCAN_PARAMETERS_SERVICE_SCAN_INTERVAL_UPDATE; 8589974af9SMilanka Ringwald little_endian_store_16(event, pos, (uint16_t) con_handle); 8689974af9SMilanka Ringwald pos += 2; 8789974af9SMilanka Ringwald little_endian_store_16(event, pos, max_scan_interval); 8889974af9SMilanka Ringwald pos += 2; 8989974af9SMilanka Ringwald little_endian_store_16(event, pos, min_scan_window); 9089974af9SMilanka Ringwald pos += 2; 9189974af9SMilanka Ringwald 92d4ac06ebSMilanka Ringwald (*scan_parameters_packet_handler)(HCI_EVENT_GATTSERVICE_META, 0, event, pos); 9389974af9SMilanka Ringwald } 9489974af9SMilanka Ringwald 95ffb44729SMilanka Ringwald static uint16_t scan_parameters_service_read_callback(hci_con_handle_t con_handle, uint16_t attribute_handle, uint16_t offset, uint8_t * buffer, uint16_t buffer_size){ 96ffb44729SMilanka Ringwald UNUSED(con_handle); 97ffb44729SMilanka Ringwald UNUSED(attribute_handle); 98ffb44729SMilanka Ringwald UNUSED(offset); 99ffb44729SMilanka Ringwald UNUSED(buffer); 100ffb44729SMilanka Ringwald UNUSED(buffer_size); 10182879dc3SMilanka Ringwald if (attribute_handle == scan_refresh_value_handle_client_configuration){ 10282879dc3SMilanka Ringwald if (buffer != NULL){ 10382879dc3SMilanka Ringwald little_endian_store_16(buffer, 0, scan_refresh_value_client_configuration); 10482879dc3SMilanka Ringwald } 10582879dc3SMilanka Ringwald return 2; 10682879dc3SMilanka Ringwald } 107ffb44729SMilanka Ringwald return 0; 108ffb44729SMilanka Ringwald } 109ffb44729SMilanka Ringwald 110ffb44729SMilanka Ringwald static int scan_parameters_service_write_callback(hci_con_handle_t con_handle, uint16_t attribute_handle, uint16_t transaction_mode, uint16_t offset, uint8_t *buffer, uint16_t buffer_size){ 111ffb44729SMilanka Ringwald UNUSED(transaction_mode); 112ffb44729SMilanka Ringwald UNUSED(offset); 113ffb44729SMilanka Ringwald UNUSED(buffer_size); 114ffb44729SMilanka Ringwald 115ffb44729SMilanka Ringwald if (attribute_handle == scan_interval_window_value_handle){ 11689974af9SMilanka Ringwald if (buffer_size == 4){ 11789974af9SMilanka Ringwald uint16_t max_scan_interval = little_endian_read_16(buffer, 0); 11889974af9SMilanka Ringwald uint16_t min_scan_window = little_endian_read_16(buffer, 2); 11989974af9SMilanka Ringwald scan_parameters_service_emit_state(con_handle, max_scan_interval, min_scan_window); 12089974af9SMilanka Ringwald return ATT_ERROR_SUCCESS; 12189974af9SMilanka Ringwald } else { 12289974af9SMilanka Ringwald return ATT_ERROR_INVALID_ATTRIBUTE_VALUE_LENGTH; 123ffb44729SMilanka Ringwald } 124ffb44729SMilanka Ringwald } 125ffb44729SMilanka Ringwald 126ffb44729SMilanka Ringwald if (attribute_handle == scan_refresh_value_handle_client_configuration){ 12789974af9SMilanka Ringwald if (buffer_size == 2){ 128ffb44729SMilanka Ringwald scan_refresh_value_client_configuration = little_endian_read_16(buffer, 0); 129ffb44729SMilanka Ringwald scan_refresh_value_client_configuration_connection = con_handle; 13089974af9SMilanka Ringwald return ATT_ERROR_SUCCESS; 13189974af9SMilanka Ringwald } else { 13289974af9SMilanka Ringwald return ATT_ERROR_INVALID_ATTRIBUTE_VALUE_LENGTH; 133ffb44729SMilanka Ringwald } 134ffb44729SMilanka Ringwald } 13589974af9SMilanka Ringwald 13689974af9SMilanka Ringwald return ATT_ERROR_UNLIKELY_ERROR; 137ffb44729SMilanka Ringwald } 138ffb44729SMilanka Ringwald 139ffb44729SMilanka Ringwald static void scan_parameters_service_refresh_can_send_now(void * context){ 14089974af9SMilanka Ringwald UNUSED(context); 14182879dc3SMilanka Ringwald const uint8_t value = 0; 14282879dc3SMilanka Ringwald att_server_notify(scan_refresh_value_client_configuration_connection, scan_refresh_value_handle, &value, 1); 143ffb44729SMilanka Ringwald } 144ffb44729SMilanka Ringwald 145ffb44729SMilanka Ringwald /** 146ffb44729SMilanka Ringwald * @brief Init Nordic SPP Service Server with ATT DB 147ffb44729SMilanka Ringwald * @param callback for tx data from peer 148ffb44729SMilanka Ringwald */ 149ffb44729SMilanka Ringwald void scan_parameters_service_server_init(btstack_packet_handler_t packet_handler){ 150ffb44729SMilanka Ringwald scan_parameters_packet_handler = packet_handler; 151ffb44729SMilanka Ringwald 152ffb44729SMilanka Ringwald // get service handle range 153ffb44729SMilanka Ringwald uint16_t start_handle = 0; 154ffb44729SMilanka Ringwald uint16_t end_handle = 0xfff; 155c436b760SMilanka Ringwald int service_found = gatt_server_get_handle_range_for_service_with_uuid16(ORG_BLUETOOTH_SERVICE_SCAN_PARAMETERS, &start_handle, &end_handle); 156ffb44729SMilanka Ringwald btstack_assert(service_found != 0); 157ffb44729SMilanka Ringwald UNUSED(service_found); 158ffb44729SMilanka Ringwald 159ffb44729SMilanka Ringwald // get characteristic value handle and client configuration handle 160ffb44729SMilanka Ringwald scan_interval_window_value_handle = gatt_server_get_value_handle_for_characteristic_with_uuid16(start_handle, end_handle, ORG_BLUETOOTH_CHARACTERISTIC_SCAN_INTERVAL_WINDOW); 161ffb44729SMilanka Ringwald scan_interval_window_value_handle_client_configuration = gatt_server_get_client_configuration_handle_for_characteristic_with_uuid16(start_handle, end_handle, ORG_BLUETOOTH_CHARACTERISTIC_SCAN_INTERVAL_WINDOW); 162ffb44729SMilanka Ringwald 163ffb44729SMilanka Ringwald scan_refresh_value_handle = gatt_server_get_value_handle_for_characteristic_with_uuid16(start_handle, end_handle, ORG_BLUETOOTH_CHARACTERISTIC_SCAN_REFRESH); 164ffb44729SMilanka Ringwald scan_refresh_value_handle_client_configuration = gatt_server_get_client_configuration_handle_for_characteristic_with_uuid16(start_handle, end_handle, ORG_BLUETOOTH_CHARACTERISTIC_SCAN_REFRESH); 165ffb44729SMilanka Ringwald 166ffb44729SMilanka Ringwald // register service with ATT Server 167ffb44729SMilanka Ringwald scan_parameters_service.start_handle = start_handle; 168ffb44729SMilanka Ringwald scan_parameters_service.end_handle = end_handle; 169ffb44729SMilanka Ringwald scan_parameters_service.read_callback = &scan_parameters_service_read_callback; 170ffb44729SMilanka Ringwald scan_parameters_service.write_callback = &scan_parameters_service_write_callback; 171ffb44729SMilanka Ringwald att_server_register_service_handler(&scan_parameters_service); 172ffb44729SMilanka Ringwald } 173ffb44729SMilanka Ringwald 174ffb44729SMilanka Ringwald 17582879dc3SMilanka Ringwald void scan_parameters_service_server_request_scan_parameters(void){ 176ffb44729SMilanka Ringwald if (scan_refresh_value_client_configuration != 0){ 177ffb44729SMilanka Ringwald scan_parameters_callback.callback = &scan_parameters_service_refresh_can_send_now; 17889974af9SMilanka Ringwald scan_parameters_callback.context = NULL; 179ffb44729SMilanka Ringwald att_server_register_can_send_now_callback(&scan_parameters_callback, scan_refresh_value_client_configuration_connection); 180ffb44729SMilanka Ringwald } 181ffb44729SMilanka Ringwald } 182