/****************************************************************************** * * Copyright 2009-2013 Broadcom Corporation * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at: * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. * ******************************************************************************/ /******************************************************************************* * * Filename: btif_gatt_server.c * * Description: GATT server implementation * ******************************************************************************/ #define LOG_TAG "bt_btif_gatt" #include #include #include #include #include #include #include #include #include "bta/include/bta_gatt_api.h" #include "bta/include/bta_sec_api.h" #include "btif/include/btif_common.h" #include "btif/include/btif_dm.h" #include "btif/include/btif_gatt.h" #include "btif/include/btif_gatt_util.h" #include "osi/include/allocator.h" #include "stack/include/bt_uuid16.h" #include "stack/include/btm_client_interface.h" #include "stack/include/main_thread.h" #include "types/ble_address_with_type.h" #include "types/bluetooth/uuid.h" #include "types/bt_transport.h" #include "types/raw_address.h" using base::Bind; using bluetooth::Uuid; using std::vector; using namespace bluetooth; namespace { tBT_TRANSPORT to_bt_transport(int val) { switch (val) { case 0: return BT_TRANSPORT_AUTO; case 1: return BT_TRANSPORT_BR_EDR; case 2: return BT_TRANSPORT_LE; default: break; } log::warn("Passed unexpected transport value:{}", val); return BT_TRANSPORT_AUTO; } } // namespace /******************************************************************************* * Constants & Macros ******************************************************************************/ #define CHECK_BTGATT_INIT() \ do { \ if (bt_gatt_callbacks == NULL) { \ log::warn("BTGATT not initialized"); \ return BT_STATUS_NOT_READY; \ } else { \ log::verbose(""); \ } \ } while (0) /******************************************************************************* * Static variables ******************************************************************************/ extern const btgatt_callbacks_t* bt_gatt_callbacks; /******************************************************************************* * Static functions ******************************************************************************/ static void btapp_gatts_copy_req_data(uint16_t event, char* p_dest, const char* p_src) { tBTA_GATTS* p_dest_data = (tBTA_GATTS*)p_dest; const tBTA_GATTS* p_src_data = (const tBTA_GATTS*)p_src; if (!p_src_data || !p_dest_data) { return; } // Copy basic structure first maybe_non_aligned_memcpy(p_dest_data, p_src_data, sizeof(*p_src_data)); // Allocate buffer for request data if necessary switch (event) { case BTA_GATTS_READ_CHARACTERISTIC_EVT: case BTA_GATTS_READ_DESCRIPTOR_EVT: case BTA_GATTS_WRITE_CHARACTERISTIC_EVT: case BTA_GATTS_WRITE_DESCRIPTOR_EVT: case BTA_GATTS_EXEC_WRITE_EVT: case BTA_GATTS_MTU_EVT: p_dest_data->req_data.p_data = (tGATTS_DATA*)osi_malloc(sizeof(tGATTS_DATA)); memcpy(p_dest_data->req_data.p_data, p_src_data->req_data.p_data, sizeof(tGATTS_DATA)); break; default: break; } } static void btapp_gatts_free_req_data(uint16_t event, tBTA_GATTS* p_data) { switch (event) { case BTA_GATTS_READ_CHARACTERISTIC_EVT: case BTA_GATTS_READ_DESCRIPTOR_EVT: case BTA_GATTS_WRITE_CHARACTERISTIC_EVT: case BTA_GATTS_WRITE_DESCRIPTOR_EVT: case BTA_GATTS_EXEC_WRITE_EVT: case BTA_GATTS_MTU_EVT: if (p_data != NULL) { osi_free_and_reset((void**)&p_data->req_data.p_data); } break; default: break; } } static void btapp_gatts_handle_cback(uint16_t event, char* p_param) { log::verbose("Event {}", event); auto callbacks = bt_gatt_callbacks; tBTA_GATTS* p_data = (tBTA_GATTS*)p_param; switch (event) { case BTA_GATTS_REG_EVT: { HAL_CBACK(callbacks, server->register_server_cb, p_data->reg_oper.status, p_data->reg_oper.server_if, p_data->reg_oper.uuid); break; } case BTA_GATTS_DEREG_EVT: break; case BTA_GATTS_CONNECT_EVT: { btif_gatt_check_encrypted_link(p_data->conn.remote_bda, p_data->conn.transport); HAL_CBACK(callbacks, server->connection_cb, static_cast(p_data->conn.conn_id), p_data->conn.server_if, true, p_data->conn.remote_bda); break; } case BTA_GATTS_DISCONNECT_EVT: { HAL_CBACK(callbacks, server->connection_cb, static_cast(p_data->conn.conn_id), p_data->conn.server_if, false, p_data->conn.remote_bda); break; } case BTA_GATTS_STOP_EVT: HAL_CBACK(callbacks, server->service_stopped_cb, p_data->srvc_oper.status, p_data->srvc_oper.server_if, p_data->srvc_oper.service_id); break; case BTA_GATTS_DELETE_EVT: HAL_CBACK(callbacks, server->service_deleted_cb, p_data->srvc_oper.status, p_data->srvc_oper.server_if, p_data->srvc_oper.service_id); break; case BTA_GATTS_READ_CHARACTERISTIC_EVT: { HAL_CBACK(callbacks, server->request_read_characteristic_cb, static_cast(p_data->req_data.conn_id), p_data->req_data.trans_id, p_data->req_data.remote_bda, p_data->req_data.p_data->read_req.handle, p_data->req_data.p_data->read_req.offset, p_data->req_data.p_data->read_req.is_long); break; } case BTA_GATTS_READ_DESCRIPTOR_EVT: { HAL_CBACK(callbacks, server->request_read_descriptor_cb, static_cast(p_data->req_data.conn_id), p_data->req_data.trans_id, p_data->req_data.remote_bda, p_data->req_data.p_data->read_req.handle, p_data->req_data.p_data->read_req.offset, p_data->req_data.p_data->read_req.is_long); break; } case BTA_GATTS_WRITE_CHARACTERISTIC_EVT: { const auto& req = p_data->req_data.p_data->write_req; HAL_CBACK(callbacks, server->request_write_characteristic_cb, static_cast(p_data->req_data.conn_id), p_data->req_data.trans_id, p_data->req_data.remote_bda, req.handle, req.offset, req.need_rsp, req.is_prep, req.value, req.len); break; } case BTA_GATTS_WRITE_DESCRIPTOR_EVT: { const auto& req = p_data->req_data.p_data->write_req; HAL_CBACK(callbacks, server->request_write_descriptor_cb, static_cast(p_data->req_data.conn_id), p_data->req_data.trans_id, p_data->req_data.remote_bda, req.handle, req.offset, req.need_rsp, req.is_prep, req.value, req.len); break; } case BTA_GATTS_EXEC_WRITE_EVT: { HAL_CBACK(callbacks, server->request_exec_write_cb, static_cast(p_data->req_data.conn_id), p_data->req_data.trans_id, p_data->req_data.remote_bda, p_data->req_data.p_data->exec_write); break; } case BTA_GATTS_CONF_EVT: HAL_CBACK(callbacks, server->indication_sent_cb, static_cast(p_data->req_data.conn_id), p_data->req_data.status); break; case BTA_GATTS_CONGEST_EVT: HAL_CBACK(callbacks, server->congestion_cb, static_cast(p_data->congest.conn_id), p_data->congest.congested); break; case BTA_GATTS_MTU_EVT: HAL_CBACK(callbacks, server->mtu_changed_cb, static_cast(p_data->req_data.conn_id), p_data->req_data.p_data->mtu); break; case BTA_GATTS_OPEN_EVT: case BTA_GATTS_CANCEL_OPEN_EVT: case BTA_GATTS_CLOSE_EVT: log::info("Empty event ({})!", event); break; case BTA_GATTS_PHY_UPDATE_EVT: HAL_CBACK(callbacks, server->phy_updated_cb, static_cast(p_data->phy_update.conn_id), p_data->phy_update.tx_phy, p_data->phy_update.rx_phy, p_data->phy_update.status); break; case BTA_GATTS_CONN_UPDATE_EVT: HAL_CBACK(callbacks, server->conn_updated_cb, static_cast(p_data->conn_update.conn_id), p_data->conn_update.interval, p_data->conn_update.latency, p_data->conn_update.timeout, p_data->conn_update.status); break; case BTA_GATTS_SUBRATE_CHG_EVT: HAL_CBACK(callbacks, server->subrate_chg_cb, static_cast(p_data->subrate_chg.conn_id), p_data->subrate_chg.subrate_factor, p_data->subrate_chg.latency, p_data->subrate_chg.cont_num, p_data->subrate_chg.timeout, p_data->subrate_chg.status); break; default: log::error("Unhandled event ({})!", event); break; } btapp_gatts_free_req_data(event, p_data); } static void btapp_gatts_cback(tBTA_GATTS_EVT event, tBTA_GATTS* p_data) { bt_status_t status; status = btif_transfer_context(btapp_gatts_handle_cback, (uint16_t)event, (char*)p_data, sizeof(tBTA_GATTS), btapp_gatts_copy_req_data); ASSERTC(status == BT_STATUS_SUCCESS, "Context transfer failed!", status); } /******************************************************************************* * Server API Functions ******************************************************************************/ static bt_status_t btif_gatts_register_app(const Uuid& bt_uuid, bool eatt_support) { CHECK_BTGATT_INIT(); return do_in_jni_thread(Bind(&BTA_GATTS_AppRegister, bt_uuid, &btapp_gatts_cback, eatt_support)); } static bt_status_t btif_gatts_unregister_app(int server_if) { CHECK_BTGATT_INIT(); return do_in_jni_thread(Bind(&BTA_GATTS_AppDeregister, server_if)); } static void btif_gatts_open_impl(int server_if, const RawAddress& address, bool is_direct, tBT_TRANSPORT transport) { // Ensure device is in inquiry database tBLE_ADDR_TYPE addr_type = BLE_ADDR_PUBLIC; int device_type = 0; if (btif_get_address_type(address, &addr_type) && btif_get_device_type(address, &device_type) && device_type != BT_DEVICE_TYPE_BREDR) { BTA_DmAddBleDevice(address, addr_type, device_type); } // Determine transport if (transport == BT_TRANSPORT_AUTO) { switch (device_type) { case BT_DEVICE_TYPE_BREDR: transport = BT_TRANSPORT_BR_EDR; break; case BT_DEVICE_TYPE_BLE: transport = BT_TRANSPORT_LE; break; case BT_DEVICE_TYPE_DUMO: transport = BT_TRANSPORT_BR_EDR; break; default: log::error("Unknown device type {}", DeviceTypeText(device_type)); // transport must not be AUTO for finding control blocks. Use LE for backward compatibility. transport = BT_TRANSPORT_LE; break; } } // Connect! BTA_GATTS_Open(server_if, address, BLE_ADDR_PUBLIC, is_direct, transport); } // Used instead of btif_gatts_open_impl if the flag // ble_gatt_server_use_address_type_in_connection is enabled. static void btif_gatts_open_impl_use_address_type(int server_if, const RawAddress& address, tBLE_ADDR_TYPE addr_type, bool is_direct, tBT_TRANSPORT transport) { int device_type = BT_DEVICE_TYPE_UNKNOWN; if (btif_get_address_type(address, &addr_type) && btif_get_device_type(address, &device_type) && device_type != BT_DEVICE_TYPE_BREDR) { BTA_DmAddBleDevice(address, addr_type, device_type); } // Determine transport if (transport == BT_TRANSPORT_AUTO) { switch (device_type) { case BT_DEVICE_TYPE_BREDR: transport = BT_TRANSPORT_BR_EDR; break; case BT_DEVICE_TYPE_BLE: case BT_DEVICE_TYPE_DUMO: transport = BT_TRANSPORT_LE; break; default: log::error("Unknown device type {}", DeviceTypeText(device_type)); // transport must not be AUTO for finding control blocks. Use LE for backward compatibility. transport = BT_TRANSPORT_LE; break; } } log::info("addr_type:{}, transport:{}", addr_type, bt_transport_text(transport)); BTA_GATTS_Open(server_if, address, addr_type, is_direct, transport); } static bt_status_t btif_gatts_open(int server_if, const RawAddress& bd_addr, uint8_t addr_type, bool is_direct, int transport) { CHECK_BTGATT_INIT(); if (com::android::bluetooth::flags::ble_gatt_server_use_address_type_in_connection()) { return do_in_jni_thread(Bind(&btif_gatts_open_impl_use_address_type, server_if, bd_addr, addr_type, is_direct, to_bt_transport(transport))); } else { return do_in_jni_thread( Bind(&btif_gatts_open_impl, server_if, bd_addr, is_direct, to_bt_transport(transport))); } } static void btif_gatts_close_impl(int server_if, const RawAddress& address, int conn_id) { // Close active connection if (conn_id != 0) { BTA_GATTS_Close(static_cast(conn_id)); } else { BTA_GATTS_CancelOpen(server_if, address, true); } // Cancel pending background connections BTA_GATTS_CancelOpen(server_if, address, false); } static bt_status_t btif_gatts_close(int server_if, const RawAddress& bd_addr, int conn_id) { CHECK_BTGATT_INIT(); return do_in_jni_thread(Bind(&btif_gatts_close_impl, server_if, bd_addr, conn_id)); } static void on_service_added_cb(tGATT_STATUS status, int server_if, vector service) { auto callbacks = bt_gatt_callbacks; HAL_CBACK(callbacks, server->service_added_cb, status, server_if, service.data(), service.size()); } static void add_service_impl(int server_if, vector service) { // TODO(jpawlowski): btif should be a pass through layer, and no checks should // be made here. This exception is added only until GATT server code is // refactored, and one can distinguish stack-internal aps from external apps if (service[0].uuid == Uuid::From16Bit(UUID_SERVCLASS_GATT_SERVER) || service[0].uuid == Uuid::From16Bit(UUID_SERVCLASS_GAP_SERVER)) { log::error("Attept to register restricted service"); auto callbacks = bt_gatt_callbacks; HAL_CBACK(callbacks, server->service_added_cb, BT_STATUS_AUTH_REJECTED, server_if, service.data(), service.size()); return; } BTA_GATTS_AddService(server_if, service, jni_thread_wrapper(base::Bind(&on_service_added_cb))); } static bt_status_t btif_gatts_add_service(int server_if, const btgatt_db_element_t* service, size_t service_count) { CHECK_BTGATT_INIT(); return do_in_jni_thread( Bind(&add_service_impl, server_if, std::vector(service, service + service_count))); } static bt_status_t btif_gatts_stop_service(int /* server_if */, int service_handle) { CHECK_BTGATT_INIT(); return do_in_jni_thread(Bind(&BTA_GATTS_StopService, service_handle)); } static bt_status_t btif_gatts_delete_service(int /* server_if */, int service_handle) { CHECK_BTGATT_INIT(); return do_in_jni_thread(Bind(&BTA_GATTS_DeleteService, service_handle)); } static bt_status_t btif_gatts_send_indication(int /* server_if */, int attribute_handle, int conn_id, int confirm, const uint8_t* value, size_t length) { CHECK_BTGATT_INIT(); if (length > GATT_MAX_ATTR_LEN) { length = GATT_MAX_ATTR_LEN; } return do_in_jni_thread(Bind(&BTA_GATTS_HandleValueIndication, static_cast(conn_id), attribute_handle, std::vector(value, value + length), confirm)); // TODO: Might need to send an ACK if handle value indication is // invoked without need for confirmation. } static void btif_gatts_send_response_impl(int conn_id, int trans_id, int status, btgatt_response_t response) { tGATTS_RSP rsp_struct; btif_to_bta_response(&rsp_struct, &response); BTA_GATTS_SendRsp(static_cast(conn_id), trans_id, static_cast(status), &rsp_struct); auto callbacks = bt_gatt_callbacks; HAL_CBACK(callbacks, server->response_confirmation_cb, 0, rsp_struct.attr_value.handle); } static bt_status_t btif_gatts_send_response(int conn_id, int trans_id, int status, const btgatt_response_t& response) { CHECK_BTGATT_INIT(); return do_in_jni_thread( Bind(&btif_gatts_send_response_impl, conn_id, trans_id, status, response)); } static bt_status_t btif_gatts_set_preferred_phy(const RawAddress& bd_addr, uint8_t tx_phy, uint8_t rx_phy, uint16_t phy_options) { CHECK_BTGATT_INIT(); do_in_main_thread(Bind( [](const RawAddress& bd_addr, uint8_t tx_phy, uint8_t rx_phy, uint16_t phy_options) { get_btm_client_interface().ble.BTM_BleSetPhy(bd_addr, tx_phy, rx_phy, phy_options); }, bd_addr, tx_phy, rx_phy, phy_options)); return BT_STATUS_SUCCESS; } static bt_status_t btif_gatts_read_phy( const RawAddress& bd_addr, base::Callback cb) { CHECK_BTGATT_INIT(); do_in_main_thread(Bind(&BTM_BleReadPhy, bd_addr, jni_thread_wrapper(cb))); return BT_STATUS_SUCCESS; } const btgatt_server_interface_t btgattServerInterface = { btif_gatts_register_app, btif_gatts_unregister_app, btif_gatts_open, btif_gatts_close, btif_gatts_add_service, btif_gatts_stop_service, btif_gatts_delete_service, btif_gatts_send_indication, btif_gatts_send_response, btif_gatts_set_preferred_phy, btif_gatts_read_phy};