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 23ffb44729SMilanka Ringwald * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL MATTHIAS 24ffb44729SMilanka Ringwald * RINGWALD 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 50ffb44729SMilanka Ringwald #include "btstack_defines.h" 51ffb44729SMilanka Ringwald #include "ble/att_db.h" 52ffb44729SMilanka Ringwald #include "ble/att_server.h" 53ffb44729SMilanka Ringwald #include "btstack_util.h" 54ffb44729SMilanka Ringwald #include "bluetooth_gatt.h" 55ffb44729SMilanka Ringwald #include "btstack_debug.h" 56ffb44729SMilanka Ringwald 57ffb44729SMilanka Ringwald #include "ble/gatt-service/scan_parameters_service_server.h" 58ffb44729SMilanka Ringwald 59ffb44729SMilanka Ringwald static btstack_packet_handler_t scan_parameters_packet_handler; 60ffb44729SMilanka Ringwald static btstack_context_callback_registration_t scan_parameters_callback; 61ffb44729SMilanka Ringwald static att_service_handler_t scan_parameters_service; 62ffb44729SMilanka Ringwald 63ffb44729SMilanka Ringwald static uint16_t scan_interval_window_value_handle; 64ffb44729SMilanka Ringwald static uint16_t scan_interval_window_value_handle_client_configuration; 65ffb44729SMilanka Ringwald 66ffb44729SMilanka Ringwald 67ffb44729SMilanka Ringwald static uint16_t scan_refresh_value; 68ffb44729SMilanka Ringwald static uint16_t scan_refresh_value_client_configuration; 69ffb44729SMilanka Ringwald static hci_con_handle_t scan_refresh_value_client_configuration_connection; 70ffb44729SMilanka Ringwald 71ffb44729SMilanka Ringwald static uint16_t scan_refresh_value_handle; 72ffb44729SMilanka Ringwald static uint16_t scan_refresh_value_handle_client_configuration; 73ffb44729SMilanka Ringwald 7489974af9SMilanka Ringwald static void scan_parameters_service_emit_state(hci_con_handle_t con_handle, uint16_t max_scan_interval, uint16_t min_scan_window){ 7589974af9SMilanka Ringwald if (scan_parameters_packet_handler == NULL) return; 7689974af9SMilanka Ringwald 7789974af9SMilanka Ringwald uint8_t event[9]; 7889974af9SMilanka Ringwald uint16_t pos = 0; 7989974af9SMilanka Ringwald event[pos++] = HCI_EVENT_GATTSERVICE_META; 8089974af9SMilanka Ringwald event[pos++] = sizeof(event) - 2; 8189974af9SMilanka Ringwald event[pos++] = GATTSERVICE_SUBEVENT_SCAN_PARAMETERS_SERVICE_SCAN_INTERVAL_UPDATE; 8289974af9SMilanka Ringwald little_endian_store_16(event, pos, (uint16_t) con_handle); 8389974af9SMilanka Ringwald pos += 2; 8489974af9SMilanka Ringwald little_endian_store_16(event, pos, max_scan_interval); 8589974af9SMilanka Ringwald pos += 2; 8689974af9SMilanka Ringwald little_endian_store_16(event, pos, min_scan_window); 8789974af9SMilanka Ringwald pos += 2; 8889974af9SMilanka Ringwald 89*d4ac06ebSMilanka Ringwald (*scan_parameters_packet_handler)(HCI_EVENT_GATTSERVICE_META, 0, event, pos); 9089974af9SMilanka Ringwald } 9189974af9SMilanka Ringwald 92ffb44729SMilanka 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){ 93ffb44729SMilanka Ringwald UNUSED(con_handle); 94ffb44729SMilanka Ringwald UNUSED(attribute_handle); 95ffb44729SMilanka Ringwald UNUSED(offset); 96ffb44729SMilanka Ringwald UNUSED(buffer); 97ffb44729SMilanka Ringwald UNUSED(buffer_size); 98ffb44729SMilanka Ringwald return 0; 99ffb44729SMilanka Ringwald } 100ffb44729SMilanka Ringwald 101ffb44729SMilanka 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){ 102ffb44729SMilanka Ringwald UNUSED(transaction_mode); 103ffb44729SMilanka Ringwald UNUSED(offset); 104ffb44729SMilanka Ringwald UNUSED(buffer_size); 105ffb44729SMilanka Ringwald 106ffb44729SMilanka Ringwald if (attribute_handle == scan_interval_window_value_handle){ 10789974af9SMilanka Ringwald if (buffer_size == 4){ 10889974af9SMilanka Ringwald uint16_t max_scan_interval = little_endian_read_16(buffer, 0); 10989974af9SMilanka Ringwald uint16_t min_scan_window = little_endian_read_16(buffer, 2); 11089974af9SMilanka Ringwald scan_parameters_service_emit_state(con_handle, max_scan_interval, min_scan_window); 11189974af9SMilanka Ringwald return ATT_ERROR_SUCCESS; 11289974af9SMilanka Ringwald } else { 11389974af9SMilanka Ringwald return ATT_ERROR_INVALID_ATTRIBUTE_VALUE_LENGTH; 114ffb44729SMilanka Ringwald } 115ffb44729SMilanka Ringwald } 116ffb44729SMilanka Ringwald 117ffb44729SMilanka Ringwald if (attribute_handle == scan_refresh_value_handle_client_configuration){ 11889974af9SMilanka Ringwald if (buffer_size == 2){ 119ffb44729SMilanka Ringwald scan_refresh_value_client_configuration = little_endian_read_16(buffer, 0); 120ffb44729SMilanka Ringwald scan_refresh_value_client_configuration_connection = con_handle; 12189974af9SMilanka Ringwald return ATT_ERROR_SUCCESS; 12289974af9SMilanka Ringwald } else { 12389974af9SMilanka Ringwald return ATT_ERROR_INVALID_ATTRIBUTE_VALUE_LENGTH; 124ffb44729SMilanka Ringwald } 125ffb44729SMilanka Ringwald } 12689974af9SMilanka Ringwald 12789974af9SMilanka Ringwald return ATT_ERROR_UNLIKELY_ERROR; 128ffb44729SMilanka Ringwald } 129ffb44729SMilanka Ringwald 130ffb44729SMilanka Ringwald static void scan_parameters_service_refresh_can_send_now(void * context){ 13189974af9SMilanka Ringwald UNUSED(context); 132ffb44729SMilanka Ringwald uint8_t value[2]; 133ffb44729SMilanka Ringwald little_endian_store_16(value, 0, scan_refresh_value); 13489974af9SMilanka Ringwald att_server_notify(scan_refresh_value_client_configuration_connection, scan_refresh_value_handle, value, 2); 135ffb44729SMilanka Ringwald } 136ffb44729SMilanka Ringwald 137ffb44729SMilanka Ringwald /** 138ffb44729SMilanka Ringwald * @brief Init Nordic SPP Service Server with ATT DB 139ffb44729SMilanka Ringwald * @param callback for tx data from peer 140ffb44729SMilanka Ringwald */ 141ffb44729SMilanka Ringwald void scan_parameters_service_server_init(btstack_packet_handler_t packet_handler){ 142ffb44729SMilanka Ringwald scan_parameters_packet_handler = packet_handler; 143ffb44729SMilanka Ringwald 144ffb44729SMilanka Ringwald // get service handle range 145ffb44729SMilanka Ringwald uint16_t start_handle = 0; 146ffb44729SMilanka Ringwald uint16_t end_handle = 0xfff; 147ffb44729SMilanka Ringwald int service_found = gatt_server_get_get_handle_range_for_service_with_uuid16(ORG_BLUETOOTH_SERVICE_SCAN_PARAMETERS, &start_handle, &end_handle); 148ffb44729SMilanka Ringwald btstack_assert(service_found != 0); 149ffb44729SMilanka Ringwald UNUSED(service_found); 150ffb44729SMilanka Ringwald 151ffb44729SMilanka Ringwald // get characteristic value handle and client configuration handle 152ffb44729SMilanka 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); 153ffb44729SMilanka 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); 154ffb44729SMilanka Ringwald 155ffb44729SMilanka Ringwald scan_refresh_value_handle = gatt_server_get_value_handle_for_characteristic_with_uuid16(start_handle, end_handle, ORG_BLUETOOTH_CHARACTERISTIC_SCAN_REFRESH); 156ffb44729SMilanka 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); 157ffb44729SMilanka Ringwald 158ffb44729SMilanka Ringwald // register service with ATT Server 159ffb44729SMilanka Ringwald scan_parameters_service.start_handle = start_handle; 160ffb44729SMilanka Ringwald scan_parameters_service.end_handle = end_handle; 161ffb44729SMilanka Ringwald scan_parameters_service.read_callback = &scan_parameters_service_read_callback; 162ffb44729SMilanka Ringwald scan_parameters_service.write_callback = &scan_parameters_service_write_callback; 163ffb44729SMilanka Ringwald att_server_register_service_handler(&scan_parameters_service); 164ffb44729SMilanka Ringwald } 165ffb44729SMilanka Ringwald 166ffb44729SMilanka Ringwald 167ffb44729SMilanka Ringwald void scan_parameters_service_server_set_scan_refresh(uint16_t scan_refresh){ 168ffb44729SMilanka Ringwald scan_refresh_value = scan_refresh; 169ffb44729SMilanka Ringwald if (scan_refresh_value_client_configuration != 0){ 170ffb44729SMilanka Ringwald scan_parameters_callback.callback = &scan_parameters_service_refresh_can_send_now; 17189974af9SMilanka Ringwald scan_parameters_callback.context = NULL; 172ffb44729SMilanka Ringwald att_server_register_can_send_now_callback(&scan_parameters_callback, scan_refresh_value_client_configuration_connection); 173ffb44729SMilanka Ringwald } 174ffb44729SMilanka Ringwald } 175