2016-09-26 21:37:39 +08:00
|
|
|
/******************************************************************************
|
|
|
|
*
|
|
|
|
* Copyright (C) 2003-2012 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.
|
|
|
|
*
|
|
|
|
******************************************************************************/
|
|
|
|
|
|
|
|
/******************************************************************************
|
|
|
|
*
|
|
|
|
* This file contains the GATT Server action functions for the state
|
|
|
|
* machine.
|
|
|
|
*
|
|
|
|
******************************************************************************/
|
|
|
|
|
|
|
|
|
2018-04-08 12:10:50 +08:00
|
|
|
#include "common/bt_target.h"
|
2016-09-26 21:37:39 +08:00
|
|
|
|
2017-04-20 07:18:05 -04:00
|
|
|
#if defined(GATTS_INCLUDED) && (GATTS_INCLUDED == TRUE)
|
2016-09-26 21:37:39 +08:00
|
|
|
|
2018-04-08 12:10:50 +08:00
|
|
|
#include "bta/utl.h"
|
|
|
|
#include "bta/bta_sys.h"
|
2016-09-26 21:37:39 +08:00
|
|
|
#include "bta_gatts_int.h"
|
2018-04-08 12:10:50 +08:00
|
|
|
#include "bta/bta_gatts_co.h"
|
|
|
|
#include "stack/btm_ble_api.h"
|
2016-09-26 21:37:39 +08:00
|
|
|
#include <string.h>
|
2018-04-08 12:10:50 +08:00
|
|
|
#include "osi/allocator.h"
|
2019-05-23 14:44:16 +08:00
|
|
|
#include "l2c_int.h"
|
2016-09-26 21:37:39 +08:00
|
|
|
|
|
|
|
static void bta_gatts_nv_save_cback(BOOLEAN is_saved, tGATTS_HNDL_RANGE *p_hndl_range);
|
|
|
|
static BOOLEAN bta_gatts_nv_srv_chg_cback(tGATTS_SRV_CHG_CMD cmd, tGATTS_SRV_CHG_REQ *p_req,
|
2016-11-25 02:10:15 +08:00
|
|
|
tGATTS_SRV_CHG_RSP *p_rsp);
|
2016-09-26 21:37:39 +08:00
|
|
|
|
|
|
|
static void bta_gatts_conn_cback (tGATT_IF gatt_if, BD_ADDR bda, UINT16 conn_id,
|
2016-11-25 02:10:15 +08:00
|
|
|
BOOLEAN connected, tGATT_DISCONN_REASON reason,
|
|
|
|
tGATT_TRANSPORT transport);
|
2016-09-26 21:37:39 +08:00
|
|
|
static void bta_gatts_send_request_cback (UINT16 conn_id,
|
2016-11-25 02:10:15 +08:00
|
|
|
UINT32 trans_id,
|
|
|
|
tGATTS_REQ_TYPE req_type, tGATTS_DATA *p_data);
|
2016-09-26 21:37:39 +08:00
|
|
|
static void bta_gatts_cong_cback (UINT16 conn_id, BOOLEAN congested);
|
2020-05-13 20:30:35 +08:00
|
|
|
extern void btc_congest_callback(tBTA_GATTS *param);
|
2016-09-26 21:37:39 +08:00
|
|
|
|
2018-11-06 15:34:38 +08:00
|
|
|
static const tGATT_CBACK bta_gatts_cback = {
|
2016-09-26 21:37:39 +08:00
|
|
|
bta_gatts_conn_cback,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
bta_gatts_send_request_cback,
|
|
|
|
NULL,
|
|
|
|
bta_gatts_cong_cback
|
|
|
|
};
|
|
|
|
|
2018-11-06 15:34:38 +08:00
|
|
|
const tGATT_APPL_INFO bta_gatts_nv_cback = {
|
2016-09-26 21:37:39 +08:00
|
|
|
bta_gatts_nv_save_cback,
|
|
|
|
bta_gatts_nv_srv_chg_cback
|
|
|
|
};
|
|
|
|
|
|
|
|
/*******************************************************************************
|
|
|
|
**
|
|
|
|
** Function bta_gatts_nv_save_cback
|
|
|
|
**
|
|
|
|
** Description NV save callback function.
|
|
|
|
**
|
|
|
|
** Parameter is_add: true is to add a handle range; otherwise is to delete.
|
|
|
|
** Returns none.
|
|
|
|
**
|
|
|
|
*******************************************************************************/
|
|
|
|
static void bta_gatts_nv_save_cback(BOOLEAN is_add, tGATTS_HNDL_RANGE *p_hndl_range)
|
|
|
|
{
|
|
|
|
bta_gatts_co_update_handle_range(is_add, (tBTA_GATTS_HNDL_RANGE *)p_hndl_range);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*******************************************************************************
|
|
|
|
**
|
|
|
|
** Function bta_gatts_nv_srv_chg_cback
|
|
|
|
**
|
|
|
|
** Description NV save callback function.
|
|
|
|
**
|
|
|
|
** Parameter is_add: true is to add a handle range; otherwise is to delete.
|
|
|
|
** Returns none.
|
|
|
|
**
|
|
|
|
*******************************************************************************/
|
|
|
|
static BOOLEAN bta_gatts_nv_srv_chg_cback(tGATTS_SRV_CHG_CMD cmd,
|
2016-11-25 02:10:15 +08:00
|
|
|
tGATTS_SRV_CHG_REQ *p_req, tGATTS_SRV_CHG_RSP *p_rsp)
|
2016-09-26 21:37:39 +08:00
|
|
|
{
|
|
|
|
return bta_gatts_co_srv_chg((tBTA_GATTS_SRV_CHG_CMD) cmd,
|
|
|
|
(tBTA_GATTS_SRV_CHG_REQ *) p_req,
|
|
|
|
(tBTA_GATTS_SRV_CHG_RSP *) p_rsp);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*******************************************************************************
|
|
|
|
**
|
|
|
|
** Function bta_gatts_enable
|
|
|
|
**
|
|
|
|
** Description enable BTA GATTS module.
|
|
|
|
**
|
|
|
|
** Returns none.
|
|
|
|
**
|
|
|
|
*******************************************************************************/
|
|
|
|
void bta_gatts_enable(tBTA_GATTS_CB *p_cb)
|
|
|
|
{
|
2016-11-25 02:10:15 +08:00
|
|
|
UINT8 index = 0;
|
2016-09-26 21:37:39 +08:00
|
|
|
tBTA_GATTS_HNDL_RANGE handle_range;
|
|
|
|
|
2016-11-25 02:10:15 +08:00
|
|
|
if (p_cb->enabled) {
|
2016-09-26 21:37:39 +08:00
|
|
|
APPL_TRACE_DEBUG("GATTS already enabled.");
|
2016-11-25 02:10:15 +08:00
|
|
|
} else {
|
2016-09-26 21:37:39 +08:00
|
|
|
memset(p_cb, 0, sizeof(tBTA_GATTS_CB));
|
|
|
|
|
|
|
|
p_cb->enabled = TRUE;
|
|
|
|
|
2016-11-25 02:10:15 +08:00
|
|
|
while ( bta_gatts_co_load_handle_range(index, &handle_range)) {
|
2016-09-26 21:37:39 +08:00
|
|
|
GATTS_AddHandleRange((tGATTS_HNDL_RANGE *)&handle_range);
|
|
|
|
memset(&handle_range, 0, sizeof(tGATTS_HNDL_RANGE));
|
|
|
|
index++;
|
|
|
|
}
|
|
|
|
|
|
|
|
APPL_TRACE_DEBUG("bta_gatts_enable: num of handle range added=%d", index);
|
|
|
|
|
2016-11-25 02:10:15 +08:00
|
|
|
if (!GATTS_NVRegister(&bta_gatts_nv_cback)) {
|
2016-09-26 21:37:39 +08:00
|
|
|
APPL_TRACE_ERROR("BTA GATTS NV register failed.");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*******************************************************************************
|
|
|
|
**
|
|
|
|
** Function bta_gatts_api_disable
|
|
|
|
**
|
|
|
|
** Description disable BTA GATTS module.
|
|
|
|
**
|
|
|
|
** Returns none.
|
|
|
|
**
|
|
|
|
*******************************************************************************/
|
|
|
|
void bta_gatts_api_disable(tBTA_GATTS_CB *p_cb)
|
|
|
|
{
|
|
|
|
UINT8 i;
|
|
|
|
|
2016-11-25 02:10:15 +08:00
|
|
|
if (p_cb->enabled) {
|
|
|
|
for (i = 0; i < BTA_GATTS_MAX_APP_NUM; i ++) {
|
|
|
|
if (p_cb->rcb[i].in_use) {
|
2016-09-26 21:37:39 +08:00
|
|
|
GATT_Deregister(p_cb->rcb[i].gatt_if);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
memset(p_cb, 0, sizeof(tBTA_GATTS_CB));
|
2016-11-25 02:10:15 +08:00
|
|
|
} else {
|
2016-09-26 21:37:39 +08:00
|
|
|
APPL_TRACE_ERROR("GATTS not enabled");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*******************************************************************************
|
|
|
|
**
|
|
|
|
** Function bta_gatts_register
|
|
|
|
**
|
|
|
|
** Description register an application.
|
|
|
|
**
|
|
|
|
** Returns none.
|
|
|
|
**
|
|
|
|
*******************************************************************************/
|
|
|
|
void bta_gatts_register(tBTA_GATTS_CB *p_cb, tBTA_GATTS_DATA *p_msg)
|
|
|
|
{
|
|
|
|
tBTA_GATTS_INT_START_IF *p_buf;
|
|
|
|
tBTA_GATTS cb_data;
|
|
|
|
tBTA_GATT_STATUS status = BTA_GATT_OK;
|
|
|
|
UINT8 i, first_unuse = 0xff;
|
|
|
|
|
2016-11-25 02:10:15 +08:00
|
|
|
if (p_cb->enabled == FALSE) {
|
2016-09-26 21:37:39 +08:00
|
|
|
bta_gatts_enable(p_cb);
|
|
|
|
}
|
|
|
|
|
2016-11-25 02:10:15 +08:00
|
|
|
for (i = 0; i < BTA_GATTS_MAX_APP_NUM; i ++) {
|
|
|
|
if (p_cb->rcb[i].in_use) {
|
|
|
|
if (bta_gatts_uuid_compare(p_cb->rcb[i].app_uuid, p_msg->api_reg.app_uuid)) {
|
2016-09-26 21:37:39 +08:00
|
|
|
APPL_TRACE_ERROR("application already registered.\n");
|
|
|
|
status = BTA_GATT_DUP_REG;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-11-25 02:10:15 +08:00
|
|
|
if (status == BTA_GATT_OK) {
|
|
|
|
for (i = 0; i < BTA_GATTS_MAX_APP_NUM; i ++) {
|
|
|
|
if (first_unuse == 0xff && !p_cb->rcb[i].in_use) {
|
2016-09-26 21:37:39 +08:00
|
|
|
first_unuse = i;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
cb_data.reg_oper.server_if = BTA_GATTS_INVALID_IF;
|
|
|
|
// btla-specific ++
|
|
|
|
memcpy(&cb_data.reg_oper.uuid, &p_msg->api_reg.app_uuid, sizeof(tBT_UUID));
|
|
|
|
// btla-specific --
|
2016-11-25 02:10:15 +08:00
|
|
|
if (first_unuse != 0xff) {
|
2016-11-16 22:11:59 +08:00
|
|
|
APPL_TRACE_VERBOSE("register application first_unuse rcb_idx = %d", first_unuse);
|
2016-09-26 21:37:39 +08:00
|
|
|
|
|
|
|
p_cb->rcb[first_unuse].in_use = TRUE;
|
|
|
|
p_cb->rcb[first_unuse].p_cback = p_msg->api_reg.p_cback;
|
|
|
|
memcpy(&p_cb->rcb[first_unuse].app_uuid, &p_msg->api_reg.app_uuid, sizeof(tBT_UUID));
|
|
|
|
cb_data.reg_oper.server_if =
|
2016-11-25 02:10:15 +08:00
|
|
|
p_cb->rcb[first_unuse].gatt_if =
|
|
|
|
GATT_Register(&p_msg->api_reg.app_uuid, &bta_gatts_cback);
|
|
|
|
if ( !p_cb->rcb[first_unuse].gatt_if) {
|
2016-09-26 21:37:39 +08:00
|
|
|
status = BTA_GATT_NO_RESOURCES;
|
2016-11-25 02:10:15 +08:00
|
|
|
} else {
|
2016-09-26 21:37:39 +08:00
|
|
|
if ((p_buf =
|
2017-08-17 21:13:45 +08:00
|
|
|
(tBTA_GATTS_INT_START_IF *) osi_malloc(sizeof(tBTA_GATTS_INT_START_IF))) != NULL) {
|
2016-09-26 21:37:39 +08:00
|
|
|
p_buf->hdr.event = BTA_GATTS_INT_START_IF_EVT;
|
|
|
|
p_buf->server_if = p_cb->rcb[first_unuse].gatt_if;
|
|
|
|
|
|
|
|
bta_sys_sendmsg(p_buf);
|
2016-11-25 02:10:15 +08:00
|
|
|
} else {
|
2016-09-26 21:37:39 +08:00
|
|
|
status = BTA_GATT_NO_RESOURCES;
|
|
|
|
memset( &p_cb->rcb[first_unuse], 0 , sizeof(tBTA_GATTS_RCB));
|
|
|
|
}
|
|
|
|
}
|
2016-11-25 02:10:15 +08:00
|
|
|
} else {
|
2016-09-26 21:37:39 +08:00
|
|
|
status = BTA_GATT_NO_RESOURCES;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
cb_data.reg_oper.status = status;
|
2016-11-25 02:10:15 +08:00
|
|
|
if (p_msg->api_reg.p_cback) {
|
2016-09-26 21:37:39 +08:00
|
|
|
(*p_msg->api_reg.p_cback)(BTA_GATTS_REG_EVT, &cb_data);
|
2016-11-25 02:10:15 +08:00
|
|
|
}
|
2016-09-26 21:37:39 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*******************************************************************************
|
|
|
|
**
|
|
|
|
** Function bta_gatts_start_if
|
|
|
|
**
|
|
|
|
** Description start an application interface.
|
|
|
|
**
|
|
|
|
** Returns none.
|
|
|
|
**
|
|
|
|
*******************************************************************************/
|
|
|
|
void bta_gatts_start_if(tBTA_GATTS_CB *p_cb, tBTA_GATTS_DATA *p_msg)
|
|
|
|
{
|
|
|
|
UNUSED(p_cb);
|
|
|
|
|
2016-11-25 02:10:15 +08:00
|
|
|
if (bta_gatts_find_app_rcb_by_app_if(p_msg->int_start_if.server_if)) {
|
2016-09-26 21:37:39 +08:00
|
|
|
GATT_StartIf(p_msg->int_start_if.server_if);
|
2016-11-25 02:10:15 +08:00
|
|
|
} else {
|
2016-09-26 21:37:39 +08:00
|
|
|
APPL_TRACE_ERROR("Unable to start app.: Unknown interface =%d",
|
2016-11-25 02:10:15 +08:00
|
|
|
p_msg->int_start_if.server_if );
|
2016-09-26 21:37:39 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
/*******************************************************************************
|
|
|
|
**
|
|
|
|
** Function bta_gatts_deregister
|
|
|
|
**
|
|
|
|
** Description deregister an application.
|
|
|
|
**
|
|
|
|
** Returns none.
|
|
|
|
**
|
|
|
|
*******************************************************************************/
|
|
|
|
void bta_gatts_deregister(tBTA_GATTS_CB *p_cb, tBTA_GATTS_DATA *p_msg)
|
|
|
|
{
|
|
|
|
tBTA_GATT_STATUS status = BTA_GATT_ERROR;
|
|
|
|
tBTA_GATTS_CBACK *p_cback = NULL;
|
|
|
|
UINT8 i;
|
|
|
|
tBTA_GATTS cb_data;
|
|
|
|
|
|
|
|
cb_data.reg_oper.server_if = p_msg->api_dereg.server_if;
|
|
|
|
cb_data.reg_oper.status = status;
|
|
|
|
|
2016-11-25 02:10:15 +08:00
|
|
|
for (i = 0; i < BTA_GATTS_MAX_APP_NUM; i ++) {
|
|
|
|
if (p_cb->rcb[i].in_use && p_cb->rcb[i].gatt_if == p_msg->api_dereg.server_if) {
|
2016-09-26 21:37:39 +08:00
|
|
|
p_cback = p_cb->rcb[i].p_cback;
|
|
|
|
status = BTA_GATT_OK;
|
|
|
|
|
|
|
|
/* deregister the app */
|
|
|
|
GATT_Deregister(p_cb->rcb[i].gatt_if);
|
|
|
|
|
|
|
|
/* reset cb */
|
|
|
|
memset(&p_cb->rcb[i], 0, sizeof(tBTA_GATTS_RCB));
|
|
|
|
cb_data.reg_oper.status = status;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-11-25 02:10:15 +08:00
|
|
|
if (p_cback) {
|
2016-09-26 21:37:39 +08:00
|
|
|
(*p_cback)(BTA_GATTS_DEREG_EVT, &cb_data);
|
2016-11-25 02:10:15 +08:00
|
|
|
} else {
|
2016-09-26 21:37:39 +08:00
|
|
|
APPL_TRACE_ERROR("application not registered.");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
/*******************************************************************************
|
|
|
|
**
|
|
|
|
** Function bta_gatts_create_srvc
|
|
|
|
**
|
|
|
|
** Description action function to create a service.
|
|
|
|
**
|
|
|
|
** Returns none.
|
|
|
|
**
|
|
|
|
*******************************************************************************/
|
2016-11-25 02:10:15 +08:00
|
|
|
void bta_gatts_create_srvc(tBTA_GATTS_CB *p_cb, tBTA_GATTS_DATA *p_msg)
|
2016-09-26 21:37:39 +08:00
|
|
|
{
|
|
|
|
UINT8 rcb_idx;
|
|
|
|
tBTA_GATTS cb_data;
|
|
|
|
UINT8 srvc_idx;
|
|
|
|
UINT16 service_id = 0;
|
|
|
|
|
|
|
|
cb_data.create.status = BTA_GATT_ERROR;
|
|
|
|
|
|
|
|
rcb_idx = bta_gatts_find_app_rcb_idx_by_app_if(p_cb, p_msg->api_create_svc.server_if);
|
|
|
|
|
2016-11-16 22:11:59 +08:00
|
|
|
APPL_TRACE_DEBUG("create service rcb_idx = %d", rcb_idx);
|
2016-09-26 21:37:39 +08:00
|
|
|
|
2016-11-25 02:10:15 +08:00
|
|
|
if (rcb_idx != BTA_GATTS_INVALID_APP) {
|
|
|
|
if ((srvc_idx = bta_gatts_alloc_srvc_cb(p_cb, rcb_idx)) != BTA_GATTS_INVALID_APP) {
|
2016-09-26 21:37:39 +08:00
|
|
|
/* create the service now */
|
|
|
|
service_id = GATTS_CreateService (p_cb->rcb[rcb_idx].gatt_if,
|
|
|
|
&p_msg->api_create_svc.service_uuid,
|
|
|
|
p_msg->api_create_svc.inst,
|
|
|
|
p_msg->api_create_svc.num_handle,
|
|
|
|
p_msg->api_create_svc.is_pri);
|
|
|
|
|
2016-11-25 02:10:15 +08:00
|
|
|
if (service_id != 0) {
|
2016-09-26 21:37:39 +08:00
|
|
|
memcpy(&p_cb->srvc_cb[srvc_idx].service_uuid,
|
2016-11-25 02:10:15 +08:00
|
|
|
&p_msg->api_create_svc.service_uuid, sizeof(tBT_UUID));
|
2016-09-26 21:37:39 +08:00
|
|
|
p_cb->srvc_cb[srvc_idx].service_id = service_id;
|
|
|
|
p_cb->srvc_cb[srvc_idx].inst_num = p_msg->api_create_svc.inst;
|
|
|
|
p_cb->srvc_cb[srvc_idx].idx = srvc_idx;
|
|
|
|
|
|
|
|
cb_data.create.status = BTA_GATT_OK;
|
|
|
|
cb_data.create.service_id = service_id;
|
2018-03-20 19:27:50 +08:00
|
|
|
|
2016-09-26 21:37:39 +08:00
|
|
|
cb_data.create.is_primary = p_msg->api_create_svc.is_pri;
|
2018-03-20 19:27:50 +08:00
|
|
|
|
2016-09-26 21:37:39 +08:00
|
|
|
cb_data.create.server_if = p_cb->rcb[rcb_idx].gatt_if;
|
2016-11-25 02:10:15 +08:00
|
|
|
} else {
|
2016-09-26 21:37:39 +08:00
|
|
|
cb_data.status = BTA_GATT_ERROR;
|
|
|
|
memset(&p_cb->srvc_cb[srvc_idx], 0, sizeof(tBTA_GATTS_SRVC_CB));
|
|
|
|
APPL_TRACE_ERROR("service creation failed.");
|
|
|
|
}
|
2018-03-20 19:27:50 +08:00
|
|
|
|
2016-09-26 21:37:39 +08:00
|
|
|
memcpy(&cb_data.create.uuid, &p_msg->api_create_svc.service_uuid, sizeof(tBT_UUID));
|
2016-11-25 02:10:15 +08:00
|
|
|
cb_data.create.svc_instance = p_msg->api_create_svc.inst;
|
2018-03-20 19:27:50 +08:00
|
|
|
|
2016-09-26 21:37:39 +08:00
|
|
|
}
|
2016-11-25 02:10:15 +08:00
|
|
|
if (p_cb->rcb[rcb_idx].p_cback) {
|
2016-09-26 21:37:39 +08:00
|
|
|
(* p_cb->rcb[rcb_idx].p_cback)(BTA_GATTS_CREATE_EVT, &cb_data);
|
2016-11-25 02:10:15 +08:00
|
|
|
}
|
|
|
|
} else { /* application not registered */
|
2016-09-26 21:37:39 +08:00
|
|
|
APPL_TRACE_ERROR("Application not registered");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
/*******************************************************************************
|
|
|
|
**
|
|
|
|
** Function bta_gatts_add_include_srvc
|
|
|
|
**
|
|
|
|
** Description action function to add an included service.
|
|
|
|
**
|
|
|
|
** Returns none.
|
|
|
|
**
|
|
|
|
*******************************************************************************/
|
2016-11-25 02:10:15 +08:00
|
|
|
void bta_gatts_add_include_srvc(tBTA_GATTS_SRVC_CB *p_srvc_cb, tBTA_GATTS_DATA *p_msg)
|
2016-09-26 21:37:39 +08:00
|
|
|
{
|
|
|
|
tBTA_GATTS_RCB *p_rcb = &bta_gatts_cb.rcb[p_srvc_cb->rcb_idx];
|
|
|
|
UINT16 attr_id = 0;
|
|
|
|
tBTA_GATTS cb_data;
|
|
|
|
|
|
|
|
attr_id = GATTS_AddIncludeService(p_msg->api_add_incl_srvc.hdr.layer_specific,
|
|
|
|
p_msg->api_add_incl_srvc.included_service_id);
|
|
|
|
|
|
|
|
cb_data.add_result.server_if = p_rcb->gatt_if;
|
|
|
|
cb_data.add_result.service_id = p_msg->api_add_incl_srvc.hdr.layer_specific;
|
|
|
|
cb_data.add_result.attr_id = attr_id;
|
|
|
|
|
2016-11-25 02:10:15 +08:00
|
|
|
if (attr_id) {
|
2016-09-26 21:37:39 +08:00
|
|
|
cb_data.add_result.status = BTA_GATT_OK;
|
2016-11-25 02:10:15 +08:00
|
|
|
} else {
|
2016-09-26 21:37:39 +08:00
|
|
|
cb_data.add_result.status = BTA_GATT_ERROR;
|
|
|
|
}
|
|
|
|
|
2016-11-25 02:10:15 +08:00
|
|
|
if (p_rcb->p_cback) {
|
2016-09-26 21:37:39 +08:00
|
|
|
(*p_rcb->p_cback)(BTA_GATTS_ADD_INCL_SRVC_EVT, &cb_data);
|
2016-11-25 02:10:15 +08:00
|
|
|
}
|
2016-09-26 21:37:39 +08:00
|
|
|
}
|
|
|
|
/*******************************************************************************
|
|
|
|
**
|
|
|
|
** Function bta_gatts_add_char
|
|
|
|
**
|
|
|
|
** Description action function to add characteristic.
|
|
|
|
**
|
|
|
|
** Returns none.
|
|
|
|
**
|
|
|
|
*******************************************************************************/
|
2016-11-25 02:10:15 +08:00
|
|
|
void bta_gatts_add_char(tBTA_GATTS_SRVC_CB *p_srvc_cb, tBTA_GATTS_DATA *p_msg)
|
2016-09-26 21:37:39 +08:00
|
|
|
{
|
|
|
|
tBTA_GATTS_RCB *p_rcb = &bta_gatts_cb.rcb[p_srvc_cb->rcb_idx];
|
|
|
|
UINT16 attr_id = 0;
|
|
|
|
tBTA_GATTS cb_data;
|
|
|
|
|
2019-03-26 14:37:37 +08:00
|
|
|
tGATT_ATTR_VAL *p_attr_val = NULL;
|
2016-12-23 11:28:47 -05:00
|
|
|
tGATTS_ATTR_CONTROL *p_control = NULL;
|
|
|
|
|
2017-01-12 00:48:48 +08:00
|
|
|
if(p_msg->api_add_char.attr_val.attr_max_len != 0){
|
|
|
|
p_attr_val = &p_msg->api_add_char.attr_val;
|
2016-12-23 11:28:47 -05:00
|
|
|
}
|
|
|
|
|
2017-01-12 00:48:48 +08:00
|
|
|
if(p_msg->api_add_char.control.auto_rsp != 0){
|
|
|
|
p_control = &p_msg->api_add_char.control;
|
2016-12-23 11:28:47 -05:00
|
|
|
}
|
2017-01-12 00:48:48 +08:00
|
|
|
|
|
|
|
|
2016-09-26 21:37:39 +08:00
|
|
|
attr_id = GATTS_AddCharacteristic(p_msg->api_add_char.hdr.layer_specific,
|
|
|
|
&p_msg->api_add_char.char_uuid,
|
|
|
|
p_msg->api_add_char.perm,
|
2016-12-23 11:28:47 -05:00
|
|
|
p_msg->api_add_char.property, p_attr_val, p_control);
|
2016-09-26 21:37:39 +08:00
|
|
|
cb_data.add_result.server_if = p_rcb->gatt_if;
|
2018-03-15 02:29:11 -04:00
|
|
|
cb_data.add_result.service_id = p_msg->api_add_char.hdr.layer_specific;
|
2016-09-26 21:37:39 +08:00
|
|
|
cb_data.add_result.attr_id = attr_id;
|
|
|
|
// btla-specific ++
|
|
|
|
memcpy(&cb_data.add_result.char_uuid, &p_msg->api_add_char.char_uuid, sizeof(tBT_UUID));
|
|
|
|
// btla-specific --
|
|
|
|
|
2016-11-25 02:10:15 +08:00
|
|
|
if (attr_id) {
|
2016-09-26 21:37:39 +08:00
|
|
|
cb_data.add_result.status = BTA_GATT_OK;
|
2016-11-25 02:10:15 +08:00
|
|
|
} else {
|
2016-09-26 21:37:39 +08:00
|
|
|
cb_data.add_result.status = BTA_GATT_ERROR;
|
|
|
|
}
|
2017-01-12 00:48:48 +08:00
|
|
|
if((p_attr_val != NULL) && (p_attr_val->attr_val != NULL)){
|
2017-08-17 21:13:45 +08:00
|
|
|
osi_free(p_attr_val->attr_val);
|
2017-01-12 00:48:48 +08:00
|
|
|
}
|
2016-09-26 21:37:39 +08:00
|
|
|
|
2016-11-25 02:10:15 +08:00
|
|
|
if (p_rcb->p_cback) {
|
2016-09-26 21:37:39 +08:00
|
|
|
(*p_rcb->p_cback)(BTA_GATTS_ADD_CHAR_EVT, &cb_data);
|
2016-11-25 02:10:15 +08:00
|
|
|
}
|
2016-09-26 21:37:39 +08:00
|
|
|
}
|
2016-12-23 11:28:47 -05:00
|
|
|
|
2016-09-26 21:37:39 +08:00
|
|
|
/*******************************************************************************
|
|
|
|
**
|
|
|
|
** Function bta_gatts_add_char_descr
|
|
|
|
**
|
|
|
|
** Description action function to add characteristic descriptor.
|
|
|
|
**
|
|
|
|
** Returns none.
|
|
|
|
**
|
|
|
|
*******************************************************************************/
|
2016-11-25 02:10:15 +08:00
|
|
|
void bta_gatts_add_char_descr(tBTA_GATTS_SRVC_CB *p_srvc_cb, tBTA_GATTS_DATA *p_msg)
|
2016-09-26 21:37:39 +08:00
|
|
|
{
|
|
|
|
tBTA_GATTS_RCB *p_rcb = &bta_gatts_cb.rcb[p_srvc_cb->rcb_idx];
|
|
|
|
UINT16 attr_id = 0;
|
|
|
|
tBTA_GATTS cb_data;
|
2016-12-23 11:28:47 -05:00
|
|
|
tGATT_ATTR_VAL *p_attr_val = NULL;
|
|
|
|
tGATTS_ATTR_CONTROL *p_control = NULL;
|
|
|
|
|
|
|
|
if (p_msg->api_add_char_descr.attr_val.attr_max_len != 0) {
|
|
|
|
p_attr_val = &p_msg->api_add_char_descr.attr_val;
|
|
|
|
}
|
2016-09-26 21:37:39 +08:00
|
|
|
|
2016-12-23 11:28:47 -05:00
|
|
|
if (p_msg->api_add_char_descr.control.auto_rsp != 0) {
|
|
|
|
p_control = &p_msg->api_add_char_descr.control;
|
|
|
|
}
|
2016-09-26 21:37:39 +08:00
|
|
|
attr_id = GATTS_AddCharDescriptor(p_msg->api_add_char_descr.hdr.layer_specific,
|
2016-11-25 02:10:15 +08:00
|
|
|
p_msg->api_add_char_descr.perm,
|
2016-12-23 11:28:47 -05:00
|
|
|
&p_msg->api_add_char_descr.descr_uuid, p_attr_val,
|
|
|
|
p_control);
|
2016-09-26 21:37:39 +08:00
|
|
|
|
|
|
|
cb_data.add_result.server_if = p_rcb->gatt_if;
|
2018-03-15 02:29:11 -04:00
|
|
|
cb_data.add_result.service_id = p_msg->api_add_char_descr.hdr.layer_specific;
|
2016-09-26 21:37:39 +08:00
|
|
|
cb_data.add_result.attr_id = attr_id;
|
|
|
|
// btla-specific ++
|
|
|
|
memcpy(&cb_data.add_result.char_uuid, &p_msg->api_add_char_descr.descr_uuid, sizeof(tBT_UUID));
|
|
|
|
// btla-specific --
|
|
|
|
|
2016-11-25 02:10:15 +08:00
|
|
|
if (attr_id) {
|
2016-09-26 21:37:39 +08:00
|
|
|
cb_data.add_result.status = BTA_GATT_OK;
|
2016-11-25 02:10:15 +08:00
|
|
|
} else {
|
2016-09-26 21:37:39 +08:00
|
|
|
cb_data.add_result.status = BTA_GATT_ERROR;
|
|
|
|
}
|
2017-01-12 00:48:48 +08:00
|
|
|
if((p_attr_val != NULL) && (p_attr_val->attr_val != NULL)){
|
2017-08-17 21:13:45 +08:00
|
|
|
osi_free(p_attr_val->attr_val);
|
2017-01-12 00:48:48 +08:00
|
|
|
}
|
2016-09-26 21:37:39 +08:00
|
|
|
|
2016-11-25 02:10:15 +08:00
|
|
|
if (p_rcb->p_cback) {
|
2016-09-26 21:37:39 +08:00
|
|
|
(*p_rcb->p_cback)(BTA_GATTS_ADD_CHAR_DESCR_EVT, &cb_data);
|
2016-11-25 02:10:15 +08:00
|
|
|
}
|
2016-09-26 21:37:39 +08:00
|
|
|
|
|
|
|
}
|
2016-12-23 11:28:47 -05:00
|
|
|
|
|
|
|
/*******************************************************************************
|
|
|
|
**
|
|
|
|
** Function bta_gatts_add_char_descr
|
|
|
|
**
|
|
|
|
** Description action function to add characteristic descriptor.
|
|
|
|
**
|
|
|
|
** Returns none.
|
|
|
|
**
|
|
|
|
*******************************************************************************/
|
|
|
|
void bta_gatts_set_attr_value(tBTA_GATTS_SRVC_CB *p_srvc_cb, tBTA_GATTS_DATA *p_msg)
|
|
|
|
{
|
|
|
|
tBTA_GATTS_RCB *p_rcb = &bta_gatts_cb.rcb[p_srvc_cb->rcb_idx];
|
2017-06-02 01:09:23 -04:00
|
|
|
UINT16 service_id = p_srvc_cb->service_id;
|
2016-12-23 11:28:47 -05:00
|
|
|
tBTA_GATTS cb_data;
|
|
|
|
tBTA_GATT_STATUS gatts_status;
|
2017-07-31 17:24:07 +08:00
|
|
|
gatts_status = GATTS_SetAttributeValue(p_msg->api_set_val.hdr.layer_specific,
|
2016-12-23 11:28:47 -05:00
|
|
|
p_msg->api_set_val.length,
|
|
|
|
p_msg->api_set_val.value);
|
|
|
|
|
|
|
|
cb_data.attr_val.server_if = p_rcb->gatt_if;
|
2017-06-02 01:09:23 -04:00
|
|
|
cb_data.attr_val.service_id = service_id;
|
|
|
|
cb_data.attr_val.attr_id = p_msg->api_set_val.hdr.layer_specific;
|
2016-12-23 11:28:47 -05:00
|
|
|
cb_data.attr_val.status = gatts_status;
|
|
|
|
|
2017-07-31 17:24:07 +08:00
|
|
|
if (p_msg->api_set_val.value != NULL){
|
2017-08-17 21:13:45 +08:00
|
|
|
osi_free(p_msg->api_set_val.value);
|
2017-07-31 17:24:07 +08:00
|
|
|
}
|
|
|
|
|
2016-12-23 11:28:47 -05:00
|
|
|
if (p_rcb->p_cback) {
|
|
|
|
(*p_rcb->p_cback)(BTA_GATTS_SET_ATTR_VAL_EVT, &cb_data);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-08-14 17:39:11 +08:00
|
|
|
tGATT_STATUS bta_gatts_get_attr_value(UINT16 attr_handle, UINT16 *length, UINT8 **value)
|
2016-12-23 11:28:47 -05:00
|
|
|
{
|
2017-08-14 17:39:11 +08:00
|
|
|
|
|
|
|
return GATTS_GetAttributeValue(attr_handle, length, value);
|
2016-12-23 11:28:47 -05:00
|
|
|
}
|
|
|
|
|
2016-09-26 21:37:39 +08:00
|
|
|
/*******************************************************************************
|
|
|
|
**
|
|
|
|
** Function bta_gatts_delete_service
|
|
|
|
**
|
|
|
|
** Description action function to delete a service.
|
|
|
|
**
|
|
|
|
** Returns none.
|
|
|
|
**
|
|
|
|
*******************************************************************************/
|
2016-11-25 02:10:15 +08:00
|
|
|
void bta_gatts_delete_service(tBTA_GATTS_SRVC_CB *p_srvc_cb, tBTA_GATTS_DATA *p_msg)
|
2016-09-26 21:37:39 +08:00
|
|
|
{
|
|
|
|
tBTA_GATTS_RCB *p_rcb = &bta_gatts_cb.rcb[p_srvc_cb->rcb_idx];
|
|
|
|
tBTA_GATTS cb_data;
|
|
|
|
|
|
|
|
cb_data.srvc_oper.server_if = p_rcb->gatt_if;
|
|
|
|
cb_data.srvc_oper.service_id = p_msg->api_add_incl_srvc.hdr.layer_specific;
|
|
|
|
|
|
|
|
if (GATTS_DeleteService(p_rcb->gatt_if,
|
|
|
|
&p_srvc_cb->service_uuid,
|
2016-11-25 02:10:15 +08:00
|
|
|
p_srvc_cb->inst_num)) {
|
2016-09-26 21:37:39 +08:00
|
|
|
cb_data.srvc_oper.status = BTA_GATT_OK;
|
|
|
|
memset(p_srvc_cb, 0, sizeof(tBTA_GATTS_SRVC_CB));
|
2016-11-25 02:10:15 +08:00
|
|
|
} else {
|
2016-09-26 21:37:39 +08:00
|
|
|
cb_data.srvc_oper.status = BTA_GATT_ERROR;
|
|
|
|
}
|
|
|
|
|
2016-11-25 02:10:15 +08:00
|
|
|
if (p_rcb->p_cback) {
|
2016-09-26 21:37:39 +08:00
|
|
|
(*p_rcb->p_cback)(BTA_GATTS_DELELTE_EVT, &cb_data);
|
2016-11-25 02:10:15 +08:00
|
|
|
}
|
2016-09-26 21:37:39 +08:00
|
|
|
|
|
|
|
}
|
|
|
|
/*******************************************************************************
|
|
|
|
**
|
|
|
|
** Function bta_gatts_start_service
|
|
|
|
**
|
|
|
|
** Description action function to start a service.
|
|
|
|
**
|
|
|
|
** Returns none.
|
|
|
|
**
|
|
|
|
*******************************************************************************/
|
2016-11-25 02:10:15 +08:00
|
|
|
void bta_gatts_start_service(tBTA_GATTS_SRVC_CB *p_srvc_cb, tBTA_GATTS_DATA *p_msg)
|
2016-09-26 21:37:39 +08:00
|
|
|
{
|
|
|
|
tBTA_GATTS_RCB *p_rcb = &bta_gatts_cb.rcb[p_srvc_cb->rcb_idx];
|
|
|
|
tBTA_GATTS cb_data;
|
|
|
|
|
|
|
|
cb_data.srvc_oper.server_if = p_rcb->gatt_if;
|
|
|
|
cb_data.srvc_oper.service_id = p_msg->api_add_incl_srvc.hdr.layer_specific;
|
|
|
|
|
|
|
|
if (GATTS_StartService(p_rcb->gatt_if,
|
|
|
|
p_srvc_cb->service_id,
|
2016-11-25 02:10:15 +08:00
|
|
|
p_msg->api_start.transport) == GATT_SUCCESS) {
|
2016-09-26 21:37:39 +08:00
|
|
|
APPL_TRACE_DEBUG("bta_gatts_start_service service_id= %d", p_srvc_cb->service_id);
|
|
|
|
cb_data.srvc_oper.status = BTA_GATT_OK;
|
2016-11-25 02:10:15 +08:00
|
|
|
} else {
|
2016-09-26 21:37:39 +08:00
|
|
|
cb_data.srvc_oper.status = BTA_GATT_ERROR;
|
|
|
|
}
|
|
|
|
|
2016-11-25 02:10:15 +08:00
|
|
|
if (p_rcb->p_cback) {
|
2016-09-26 21:37:39 +08:00
|
|
|
(*p_rcb->p_cback)(BTA_GATTS_START_EVT, &cb_data);
|
2016-11-25 02:10:15 +08:00
|
|
|
}
|
2016-09-26 21:37:39 +08:00
|
|
|
|
|
|
|
}
|
|
|
|
/*******************************************************************************
|
|
|
|
**
|
|
|
|
** Function bta_gatts_stop_service
|
|
|
|
**
|
|
|
|
** Description action function to stop a service.
|
|
|
|
**
|
|
|
|
** Returns none.
|
|
|
|
**
|
|
|
|
*******************************************************************************/
|
2016-11-25 02:10:15 +08:00
|
|
|
void bta_gatts_stop_service(tBTA_GATTS_SRVC_CB *p_srvc_cb, tBTA_GATTS_DATA *p_msg)
|
2016-09-26 21:37:39 +08:00
|
|
|
{
|
|
|
|
tBTA_GATTS_RCB *p_rcb = &bta_gatts_cb.rcb[p_srvc_cb->rcb_idx];
|
|
|
|
tBTA_GATTS cb_data;
|
|
|
|
UNUSED(p_msg);
|
|
|
|
|
|
|
|
GATTS_StopService(p_srvc_cb->service_id);
|
|
|
|
cb_data.srvc_oper.server_if = p_rcb->gatt_if;
|
|
|
|
cb_data.srvc_oper.service_id = p_srvc_cb->service_id;
|
|
|
|
cb_data.srvc_oper.status = BTA_GATT_OK;
|
2017-06-14 18:42:15 +08:00
|
|
|
APPL_TRACE_DEBUG("bta_gatts_stop_service service_id= %d", p_srvc_cb->service_id);
|
2016-09-26 21:37:39 +08:00
|
|
|
|
2016-11-25 02:10:15 +08:00
|
|
|
if (p_rcb->p_cback) {
|
2016-09-26 21:37:39 +08:00
|
|
|
(*p_rcb->p_cback)(BTA_GATTS_STOP_EVT, &cb_data);
|
2016-11-25 02:10:15 +08:00
|
|
|
}
|
2016-09-26 21:37:39 +08:00
|
|
|
|
|
|
|
}
|
|
|
|
/*******************************************************************************
|
|
|
|
**
|
|
|
|
** Function bta_gatts_send_rsp
|
|
|
|
**
|
|
|
|
** Description GATTS send response.
|
|
|
|
**
|
|
|
|
** Returns none.
|
|
|
|
**
|
|
|
|
*******************************************************************************/
|
2016-11-25 02:10:15 +08:00
|
|
|
void bta_gatts_send_rsp (tBTA_GATTS_CB *p_cb, tBTA_GATTS_DATA *p_msg)
|
2016-09-26 21:37:39 +08:00
|
|
|
{
|
|
|
|
UNUSED(p_cb);
|
|
|
|
|
|
|
|
if (GATTS_SendRsp (p_msg->api_rsp.hdr.layer_specific,
|
2016-11-25 02:10:15 +08:00
|
|
|
p_msg->api_rsp.trans_id,
|
|
|
|
p_msg->api_rsp.status,
|
|
|
|
(tGATTS_RSP *)p_msg->api_rsp.p_rsp) != GATT_SUCCESS) {
|
2016-10-15 05:46:10 -04:00
|
|
|
APPL_TRACE_ERROR("Sending response failed\n");
|
2016-09-26 21:37:39 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
/*******************************************************************************
|
|
|
|
**
|
|
|
|
** Function bta_gatts_indicate_handle
|
|
|
|
**
|
|
|
|
** Description GATTS send handle value indication or notification.
|
|
|
|
**
|
|
|
|
** Returns none.
|
|
|
|
**
|
|
|
|
*******************************************************************************/
|
2016-11-25 02:10:15 +08:00
|
|
|
void bta_gatts_indicate_handle (tBTA_GATTS_CB *p_cb, tBTA_GATTS_DATA *p_msg)
|
2016-09-26 21:37:39 +08:00
|
|
|
{
|
|
|
|
tBTA_GATTS_SRVC_CB *p_srvc_cb;
|
|
|
|
tBTA_GATTS_RCB *p_rcb = NULL;
|
|
|
|
tBTA_GATT_STATUS status = BTA_GATT_ILLEGAL_PARAMETER;
|
|
|
|
tGATT_IF gatt_if;
|
|
|
|
BD_ADDR remote_bda;
|
|
|
|
tBTA_TRANSPORT transport;
|
|
|
|
tBTA_GATTS cb_data;
|
|
|
|
|
|
|
|
p_srvc_cb = bta_gatts_find_srvc_cb_by_attr_id (p_cb, p_msg->api_indicate.attr_id);
|
|
|
|
|
2016-11-25 02:10:15 +08:00
|
|
|
if (p_srvc_cb ) {
|
2016-09-26 21:37:39 +08:00
|
|
|
if (GATT_GetConnectionInfor(p_msg->api_indicate.hdr.layer_specific,
|
2016-11-25 02:10:15 +08:00
|
|
|
&gatt_if, remote_bda, &transport)) {
|
2016-09-26 21:37:39 +08:00
|
|
|
p_rcb = bta_gatts_find_app_rcb_by_app_if(gatt_if);
|
|
|
|
|
2019-03-26 14:37:37 +08:00
|
|
|
if (p_msg->api_indicate.need_confirm) {
|
2016-09-26 21:37:39 +08:00
|
|
|
|
|
|
|
status = GATTS_HandleValueIndication (p_msg->api_indicate.hdr.layer_specific,
|
|
|
|
p_msg->api_indicate.attr_id,
|
|
|
|
p_msg->api_indicate.len,
|
|
|
|
p_msg->api_indicate.value);
|
2019-03-26 14:37:37 +08:00
|
|
|
} else {
|
2020-05-11 19:24:09 +08:00
|
|
|
l2ble_update_att_acl_pkt_num(L2CA_DECREASE_BTU_NUM, NULL);
|
2016-09-26 21:37:39 +08:00
|
|
|
status = GATTS_HandleValueNotification (p_msg->api_indicate.hdr.layer_specific,
|
|
|
|
p_msg->api_indicate.attr_id,
|
|
|
|
p_msg->api_indicate.len,
|
|
|
|
p_msg->api_indicate.value);
|
2019-03-26 14:37:37 +08:00
|
|
|
}
|
2016-09-26 21:37:39 +08:00
|
|
|
|
|
|
|
/* if over BR_EDR, inform PM for mode change */
|
2016-11-25 02:10:15 +08:00
|
|
|
if (transport == BTA_TRANSPORT_BR_EDR) {
|
2016-09-26 21:37:39 +08:00
|
|
|
bta_sys_busy(BTA_ID_GATTS, BTA_ALL_APP_ID, remote_bda);
|
|
|
|
bta_sys_idle(BTA_ID_GATTS, BTA_ALL_APP_ID, remote_bda);
|
|
|
|
}
|
2016-11-25 02:10:15 +08:00
|
|
|
} else {
|
2016-09-26 21:37:39 +08:00
|
|
|
APPL_TRACE_ERROR("Unknown connection ID: %d fail sending notification",
|
2016-11-25 02:10:15 +08:00
|
|
|
p_msg->api_indicate.hdr.layer_specific);
|
2016-09-26 21:37:39 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if ((status != GATT_SUCCESS || !p_msg->api_indicate.need_confirm) &&
|
2016-11-25 02:10:15 +08:00
|
|
|
p_rcb && p_cb->rcb[p_srvc_cb->rcb_idx].p_cback) {
|
2016-09-26 21:37:39 +08:00
|
|
|
cb_data.req_data.status = status;
|
|
|
|
cb_data.req_data.conn_id = p_msg->api_indicate.hdr.layer_specific;
|
2018-11-06 21:01:54 +08:00
|
|
|
cb_data.req_data.value = NULL;
|
|
|
|
cb_data.req_data.data_len = 0;
|
2018-10-16 14:38:49 +08:00
|
|
|
cb_data.req_data.handle = p_msg->api_indicate.attr_id;
|
2016-09-26 21:37:39 +08:00
|
|
|
|
2023-02-06 11:34:31 +07:00
|
|
|
if (p_msg->api_indicate.len > 0) {
|
2018-11-06 21:01:54 +08:00
|
|
|
cb_data.req_data.value = (uint8_t *) osi_malloc(p_msg->api_indicate.len);
|
|
|
|
if (cb_data.req_data.value != NULL) {
|
|
|
|
memset(cb_data.req_data.value, 0, p_msg->api_indicate.len);
|
|
|
|
cb_data.req_data.data_len = p_msg->api_indicate.len;
|
|
|
|
memcpy(cb_data.req_data.value, p_msg->api_indicate.value, p_msg->api_indicate.len);
|
|
|
|
} else {
|
|
|
|
APPL_TRACE_ERROR("%s, malloc failed", __func__);
|
|
|
|
}
|
|
|
|
} else {
|
2023-02-06 11:34:31 +07:00
|
|
|
APPL_TRACE_ERROR("%s, incorrect length", __func__);
|
2017-10-26 14:39:49 +08:00
|
|
|
}
|
2016-09-26 21:37:39 +08:00
|
|
|
(*p_rcb->p_cback)(BTA_GATTS_CONF_EVT, &cb_data);
|
2018-08-08 14:16:03 +05:30
|
|
|
if (cb_data.req_data.value != NULL) {
|
|
|
|
osi_free(cb_data.req_data.value);
|
|
|
|
cb_data.req_data.value = NULL;
|
|
|
|
}
|
2016-09-26 21:37:39 +08:00
|
|
|
}
|
2016-11-25 02:10:15 +08:00
|
|
|
} else {
|
2016-09-26 21:37:39 +08:00
|
|
|
APPL_TRACE_ERROR("Not an registered servce attribute ID: 0x%04x",
|
2016-11-25 02:10:15 +08:00
|
|
|
p_msg->api_indicate.attr_id);
|
2016-09-26 21:37:39 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*******************************************************************************
|
|
|
|
**
|
|
|
|
** Function bta_gatts_open
|
|
|
|
**
|
|
|
|
** Description
|
|
|
|
**
|
|
|
|
** Returns none.
|
|
|
|
**
|
|
|
|
*******************************************************************************/
|
2016-11-25 02:10:15 +08:00
|
|
|
void bta_gatts_open (tBTA_GATTS_CB *p_cb, tBTA_GATTS_DATA *p_msg)
|
2016-09-26 21:37:39 +08:00
|
|
|
{
|
2016-11-25 02:10:15 +08:00
|
|
|
tBTA_GATTS_RCB *p_rcb = NULL;
|
|
|
|
tBTA_GATT_STATUS status = BTA_GATT_ERROR;
|
2016-09-26 21:37:39 +08:00
|
|
|
UINT16 conn_id;
|
2017-06-01 14:16:24 +08:00
|
|
|
tBTA_GATTS_OPEN open;
|
2016-09-26 21:37:39 +08:00
|
|
|
UNUSED(p_cb);
|
|
|
|
|
2016-11-25 02:10:15 +08:00
|
|
|
if ((p_rcb = bta_gatts_find_app_rcb_by_app_if(p_msg->api_open.server_if)) != NULL) {
|
2016-09-26 21:37:39 +08:00
|
|
|
/* should always get the connection ID */
|
2018-03-23 11:08:03 +08:00
|
|
|
if (GATT_Connect(p_rcb->gatt_if, p_msg->api_open.remote_bda, BLE_ADDR_UNKNOWN_TYPE,
|
2020-07-09 20:58:13 +08:00
|
|
|
p_msg->api_open.is_direct, p_msg->api_open.transport, FALSE)) {
|
2016-09-26 21:37:39 +08:00
|
|
|
status = BTA_GATT_OK;
|
|
|
|
|
|
|
|
if (GATT_GetConnIdIfConnected(p_rcb->gatt_if, p_msg->api_open.remote_bda,
|
2016-11-25 02:10:15 +08:00
|
|
|
&conn_id, p_msg->api_open.transport)) {
|
2016-09-26 21:37:39 +08:00
|
|
|
status = BTA_GATT_ALREADY_OPEN;
|
|
|
|
}
|
|
|
|
}
|
2016-11-25 02:10:15 +08:00
|
|
|
} else {
|
2016-09-26 21:37:39 +08:00
|
|
|
APPL_TRACE_ERROR("Inavlide server_if=%d", p_msg->api_open.server_if);
|
|
|
|
}
|
|
|
|
|
2016-11-25 02:10:15 +08:00
|
|
|
if (p_rcb && p_rcb->p_cback) {
|
2017-06-01 14:16:24 +08:00
|
|
|
open.status = status;
|
|
|
|
open.server_if = p_msg->api_open.server_if;
|
|
|
|
(*p_rcb->p_cback)(BTA_GATTS_OPEN_EVT, (tBTA_GATTS *)&open);
|
2016-11-25 02:10:15 +08:00
|
|
|
}
|
2016-09-26 21:37:39 +08:00
|
|
|
|
|
|
|
}
|
|
|
|
/*******************************************************************************
|
|
|
|
**
|
|
|
|
** Function bta_gatts_cancel_open
|
|
|
|
**
|
|
|
|
** Description
|
|
|
|
**
|
|
|
|
** Returns none.
|
|
|
|
**
|
|
|
|
*******************************************************************************/
|
2016-11-25 02:10:15 +08:00
|
|
|
void bta_gatts_cancel_open (tBTA_GATTS_CB *p_cb, tBTA_GATTS_DATA *p_msg)
|
2016-09-26 21:37:39 +08:00
|
|
|
{
|
|
|
|
tBTA_GATTS_RCB *p_rcb;
|
2016-11-25 02:10:15 +08:00
|
|
|
tBTA_GATT_STATUS status = BTA_GATT_ERROR;
|
2017-06-01 14:16:24 +08:00
|
|
|
tBTA_GATTS_CANCEL_OPEN cancel_open;
|
2016-09-26 21:37:39 +08:00
|
|
|
UNUSED(p_cb);
|
|
|
|
|
2016-11-25 02:10:15 +08:00
|
|
|
if ((p_rcb = bta_gatts_find_app_rcb_by_app_if(p_msg->api_cancel_open.server_if)) != NULL) {
|
2016-09-26 21:37:39 +08:00
|
|
|
if (!GATT_CancelConnect(p_rcb->gatt_if, p_msg->api_cancel_open.remote_bda,
|
2016-11-25 02:10:15 +08:00
|
|
|
p_msg->api_cancel_open.is_direct)) {
|
2016-09-26 21:37:39 +08:00
|
|
|
APPL_TRACE_ERROR("bta_gatts_cancel_open failed for open request");
|
2016-11-25 02:10:15 +08:00
|
|
|
} else {
|
|
|
|
status = BTA_GATT_OK;
|
2016-09-26 21:37:39 +08:00
|
|
|
}
|
2016-11-25 02:10:15 +08:00
|
|
|
} else {
|
2016-09-26 21:37:39 +08:00
|
|
|
APPL_TRACE_ERROR("Inavlide server_if=%d", p_msg->api_cancel_open.server_if);
|
|
|
|
}
|
|
|
|
|
2016-11-25 02:10:15 +08:00
|
|
|
if (p_rcb && p_rcb->p_cback) {
|
2017-06-01 14:16:24 +08:00
|
|
|
cancel_open.status = status;
|
|
|
|
cancel_open.server_if = p_msg->api_cancel_open.server_if;
|
|
|
|
(*p_rcb->p_cback)(BTA_GATTS_CANCEL_OPEN_EVT, (tBTA_GATTS *)&cancel_open);
|
|
|
|
|
2016-11-25 02:10:15 +08:00
|
|
|
}
|
2016-09-26 21:37:39 +08:00
|
|
|
}
|
|
|
|
/*******************************************************************************
|
|
|
|
**
|
|
|
|
** Function bta_gatts_close
|
|
|
|
**
|
|
|
|
** Description
|
|
|
|
**
|
|
|
|
** Returns none.
|
|
|
|
**
|
|
|
|
*******************************************************************************/
|
2016-11-25 02:10:15 +08:00
|
|
|
void bta_gatts_close (tBTA_GATTS_CB *p_cb, tBTA_GATTS_DATA *p_msg)
|
2016-09-26 21:37:39 +08:00
|
|
|
{
|
|
|
|
tBTA_GATTS_RCB *p_rcb;
|
2016-11-25 02:10:15 +08:00
|
|
|
tBTA_GATT_STATUS status = BTA_GATT_ERROR;
|
2016-09-26 21:37:39 +08:00
|
|
|
tGATT_IF gatt_if;
|
|
|
|
BD_ADDR remote_bda;
|
|
|
|
tBTA_GATT_TRANSPORT transport;
|
2017-06-01 14:16:24 +08:00
|
|
|
tBTA_GATTS_CLOSE close;
|
2016-09-26 21:37:39 +08:00
|
|
|
UNUSED(p_cb);
|
|
|
|
|
2016-11-25 02:10:15 +08:00
|
|
|
if (GATT_GetConnectionInfor(p_msg->hdr.layer_specific, &gatt_if, remote_bda, &transport)) {
|
|
|
|
if (GATT_Disconnect(p_msg->hdr.layer_specific) != GATT_SUCCESS) {
|
2016-09-26 21:37:39 +08:00
|
|
|
APPL_TRACE_ERROR("bta_gatts_close fail conn_id=%d", p_msg->hdr.layer_specific);
|
2016-11-25 02:10:15 +08:00
|
|
|
} else {
|
|
|
|
status = BTA_GATT_OK;
|
2016-09-26 21:37:39 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
p_rcb = bta_gatts_find_app_rcb_by_app_if(gatt_if);
|
|
|
|
|
2016-11-25 02:10:15 +08:00
|
|
|
if (p_rcb && p_rcb->p_cback) {
|
|
|
|
if (transport == BTA_TRANSPORT_BR_EDR) {
|
|
|
|
bta_sys_conn_close( BTA_ID_GATTS , BTA_ALL_APP_ID, remote_bda);
|
|
|
|
}
|
2016-09-26 21:37:39 +08:00
|
|
|
|
2017-06-01 14:16:24 +08:00
|
|
|
close.status = status;
|
|
|
|
close.conn_id = p_msg->hdr.layer_specific;
|
|
|
|
(*p_rcb->p_cback)(BTA_GATTS_CLOSE_EVT, (tBTA_GATTS *)&close);
|
2016-09-26 21:37:39 +08:00
|
|
|
}
|
2016-11-25 02:10:15 +08:00
|
|
|
} else {
|
2016-09-26 21:37:39 +08:00
|
|
|
APPL_TRACE_ERROR("Unknown connection ID: %d", p_msg->hdr.layer_specific);
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
2018-09-18 16:20:13 +08:00
|
|
|
|
|
|
|
/*******************************************************************************
|
|
|
|
**
|
|
|
|
** Function bta_gatts_send_service_change_indication
|
|
|
|
**
|
|
|
|
** Description gatts send service change indication
|
|
|
|
**
|
|
|
|
** Returns none.
|
|
|
|
**
|
|
|
|
*******************************************************************************/
|
|
|
|
void bta_gatts_send_service_change_indication (tBTA_GATTS_DATA *p_msg)
|
|
|
|
{
|
|
|
|
tBTA_GATTS_RCB *p_rcb = bta_gatts_find_app_rcb_by_app_if(p_msg->api_send_service_change.server_if);
|
|
|
|
tBTA_GATTS_SERVICE_CHANGE service_change;
|
|
|
|
tBTA_GATT_STATUS status = BTA_GATT_OK;
|
|
|
|
UINT16 addr[BD_ADDR_LEN] = {0};
|
|
|
|
if(memcmp(p_msg->api_send_service_change.remote_bda, addr, BD_ADDR_LEN) != 0) {
|
|
|
|
BD_ADDR bd_addr;
|
|
|
|
memcpy(bd_addr, p_msg->api_send_service_change.remote_bda, BD_ADDR_LEN);
|
|
|
|
status = GATT_SendServiceChangeIndication(bd_addr);
|
|
|
|
} else {
|
2019-03-26 14:37:37 +08:00
|
|
|
status = GATT_SendServiceChangeIndication(NULL);
|
2018-09-18 16:20:13 +08:00
|
|
|
}
|
|
|
|
if (p_rcb && p_rcb->p_cback) {
|
|
|
|
service_change.status = status;
|
|
|
|
service_change.server_if = p_msg->api_send_service_change.server_if;
|
|
|
|
(*p_rcb->p_cback)(BTA_GATTS_SEND_SERVICE_CHANGE_EVT, (tBTA_GATTS *)&service_change);
|
2019-03-26 14:37:37 +08:00
|
|
|
}
|
2018-09-18 16:20:13 +08:00
|
|
|
}
|
|
|
|
|
2016-09-26 21:37:39 +08:00
|
|
|
/*******************************************************************************
|
|
|
|
**
|
|
|
|
** Function bta_gatts_listen
|
|
|
|
**
|
|
|
|
** Description Start or stop listening for LE connection on a GATT server
|
|
|
|
**
|
|
|
|
** Returns none.
|
|
|
|
**
|
|
|
|
*******************************************************************************/
|
2016-11-25 02:10:15 +08:00
|
|
|
void bta_gatts_listen(tBTA_GATTS_CB *p_cb, tBTA_GATTS_DATA *p_msg)
|
2016-09-26 21:37:39 +08:00
|
|
|
{
|
|
|
|
tBTA_GATTS_RCB *p_rcb = bta_gatts_find_app_rcb_by_app_if(p_msg->api_listen.server_if);
|
|
|
|
tBTA_GATTS cb_data;
|
|
|
|
UNUSED(p_cb);
|
|
|
|
|
|
|
|
cb_data.reg_oper.status = BTA_GATT_OK;
|
|
|
|
cb_data.reg_oper.server_if = p_msg->api_listen.server_if;
|
|
|
|
|
2016-11-25 02:10:15 +08:00
|
|
|
if (p_rcb == NULL) {
|
2016-09-26 21:37:39 +08:00
|
|
|
APPL_TRACE_ERROR("Unknown GATTS application");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!GATT_Listen(p_msg->api_listen.server_if,
|
|
|
|
p_msg->api_listen.start,
|
2016-11-25 02:10:15 +08:00
|
|
|
p_msg->api_listen.remote_bda)) {
|
2016-09-26 21:37:39 +08:00
|
|
|
cb_data.status = BTA_GATT_ERROR;
|
|
|
|
APPL_TRACE_ERROR("bta_gatts_listen Listen failed");
|
|
|
|
}
|
|
|
|
|
2016-11-25 02:10:15 +08:00
|
|
|
if (p_rcb->p_cback) {
|
2016-09-26 21:37:39 +08:00
|
|
|
(*p_rcb->p_cback)(BTA_GATTS_LISTEN_EVT, &cb_data);
|
2016-11-25 02:10:15 +08:00
|
|
|
}
|
2016-09-26 21:37:39 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/*******************************************************************************
|
|
|
|
**
|
|
|
|
** Function bta_gatts_request_cback
|
|
|
|
**
|
|
|
|
** Description GATTS attribute request callback.
|
|
|
|
**
|
|
|
|
** Returns none.
|
|
|
|
**
|
|
|
|
*******************************************************************************/
|
|
|
|
static void bta_gatts_send_request_cback (UINT16 conn_id,
|
2016-11-25 02:10:15 +08:00
|
|
|
UINT32 trans_id,
|
|
|
|
tGATTS_REQ_TYPE req_type, tGATTS_DATA *p_data)
|
2016-09-26 21:37:39 +08:00
|
|
|
{
|
|
|
|
tBTA_GATTS cb_data;
|
|
|
|
tBTA_GATTS_RCB *p_rcb;
|
|
|
|
tGATT_IF gatt_if;
|
|
|
|
tBTA_GATT_TRANSPORT transport;
|
|
|
|
|
|
|
|
memset(&cb_data, 0 , sizeof(tBTA_GATTS));
|
|
|
|
|
2016-11-25 02:10:15 +08:00
|
|
|
if (GATT_GetConnectionInfor(conn_id, &gatt_if, cb_data.req_data.remote_bda, &transport)) {
|
2016-09-26 21:37:39 +08:00
|
|
|
p_rcb = bta_gatts_find_app_rcb_by_app_if(gatt_if);
|
|
|
|
|
|
|
|
APPL_TRACE_DEBUG ("bta_gatts_send_request_cback conn_id=%d trans_id=%d req_type=%d",
|
2016-11-25 02:10:15 +08:00
|
|
|
conn_id, trans_id, req_type);
|
2016-09-26 21:37:39 +08:00
|
|
|
|
2016-11-25 02:10:15 +08:00
|
|
|
if (p_rcb && p_rcb->p_cback) {
|
2016-09-26 21:37:39 +08:00
|
|
|
/* if over BR_EDR, inform PM for mode change */
|
2016-11-25 02:10:15 +08:00
|
|
|
if (transport == BTA_TRANSPORT_BR_EDR) {
|
2016-09-26 21:37:39 +08:00
|
|
|
bta_sys_busy(BTA_ID_GATTS, BTA_ALL_APP_ID, cb_data.req_data.remote_bda);
|
|
|
|
bta_sys_idle(BTA_ID_GATTS, BTA_ALL_APP_ID, cb_data.req_data.remote_bda);
|
|
|
|
}
|
|
|
|
|
|
|
|
cb_data.req_data.conn_id = conn_id;
|
|
|
|
cb_data.req_data.trans_id = trans_id;
|
|
|
|
cb_data.req_data.p_data = (tBTA_GATTS_REQ_DATA *)p_data;
|
|
|
|
|
2018-12-24 14:35:24 +08:00
|
|
|
if(req_type == BTA_GATTS_CONF_EVT) {
|
2019-03-26 14:37:37 +08:00
|
|
|
cb_data.req_data.handle = p_data->handle;
|
2018-12-24 14:35:24 +08:00
|
|
|
}
|
2016-09-26 21:37:39 +08:00
|
|
|
(*p_rcb->p_cback)(req_type, &cb_data);
|
2016-11-25 02:10:15 +08:00
|
|
|
} else {
|
2016-09-26 21:37:39 +08:00
|
|
|
APPL_TRACE_ERROR("connection request on gatt_if[%d] is not interested", gatt_if);
|
|
|
|
}
|
2016-11-25 02:10:15 +08:00
|
|
|
} else {
|
2016-11-28 14:58:30 +08:00
|
|
|
APPL_TRACE_ERROR("request received on unknown connection ID: %d", conn_id);
|
2016-09-26 21:37:39 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*******************************************************************************
|
|
|
|
**
|
|
|
|
** Function bta_gatts_conn_cback
|
|
|
|
**
|
|
|
|
** Description connection callback.
|
|
|
|
**
|
|
|
|
** Returns none.
|
|
|
|
**
|
|
|
|
*******************************************************************************/
|
|
|
|
static void bta_gatts_conn_cback (tGATT_IF gatt_if, BD_ADDR bda, UINT16 conn_id,
|
|
|
|
BOOLEAN connected, tGATT_DISCONN_REASON reason,
|
|
|
|
tGATT_TRANSPORT transport)
|
|
|
|
{
|
2019-05-23 14:44:16 +08:00
|
|
|
tBTA_GATTS cb_data = {0};
|
2016-11-25 02:10:15 +08:00
|
|
|
UINT8 evt = connected ? BTA_GATTS_CONNECT_EVT : BTA_GATTS_DISCONNECT_EVT;
|
2016-09-26 21:37:39 +08:00
|
|
|
tBTA_GATTS_RCB *p_reg;
|
|
|
|
|
|
|
|
APPL_TRACE_DEBUG ("bta_gatts_conn_cback gatt_if=%d conn_id=%d connected=%d reason = 0x%04d",
|
2016-11-25 02:10:15 +08:00
|
|
|
gatt_if, conn_id, connected, reason);
|
2016-09-26 21:37:39 +08:00
|
|
|
APPL_TRACE_DEBUG("bta_gatts_conn_cback bda :%02x-%02x-%02x-%02x-%02x-%02x ",
|
2016-11-25 02:10:15 +08:00
|
|
|
bda[0], bda[1], bda[2], bda[3], bda[4], bda[5]);
|
2016-09-26 21:37:39 +08:00
|
|
|
|
|
|
|
bt_bdaddr_t bdaddr;
|
|
|
|
bdcpy(bdaddr.address, bda);
|
|
|
|
/*
|
|
|
|
if (connected)
|
|
|
|
btif_debug_conn_state(bdaddr, BTIF_DEBUG_CONNECTED, GATT_CONN_UNKNOWN);
|
|
|
|
else
|
|
|
|
btif_debug_conn_state(bdaddr, BTIF_DEBUG_DISCONNECTED, reason);
|
|
|
|
*/
|
|
|
|
p_reg = bta_gatts_find_app_rcb_by_app_if(gatt_if);
|
|
|
|
|
2016-11-25 02:10:15 +08:00
|
|
|
if (p_reg && p_reg->p_cback) {
|
2016-09-26 21:37:39 +08:00
|
|
|
/* there is no RM for GATT */
|
2016-11-25 02:10:15 +08:00
|
|
|
if (transport == BTA_TRANSPORT_BR_EDR) {
|
|
|
|
if (connected) {
|
2016-09-26 21:37:39 +08:00
|
|
|
bta_sys_conn_open(BTA_ID_GATTS, BTA_ALL_APP_ID, bda);
|
2016-11-25 02:10:15 +08:00
|
|
|
} else {
|
|
|
|
bta_sys_conn_close( BTA_ID_GATTS , BTA_ALL_APP_ID, bda);
|
|
|
|
}
|
2016-09-26 21:37:39 +08:00
|
|
|
}
|
2019-05-23 14:44:16 +08:00
|
|
|
if(evt == BTA_GATTS_CONNECT_EVT) {
|
|
|
|
tL2C_LCB *p_lcb = l2cu_find_lcb_by_bd_addr(bda, BT_TRANSPORT_LE);
|
|
|
|
if(p_lcb != NULL) {
|
|
|
|
cb_data.conn.conn_params.interval = p_lcb->current_used_conn_interval;
|
|
|
|
cb_data.conn.conn_params.latency = p_lcb->current_used_conn_latency;
|
|
|
|
cb_data.conn.conn_params.timeout = p_lcb->current_used_conn_timeout;
|
2020-08-26 17:03:39 +08:00
|
|
|
cb_data.conn.link_role = p_lcb->link_role;
|
2023-02-01 12:08:05 +08:00
|
|
|
#if (BLE_INCLUDED == TRUE)
|
|
|
|
cb_data.conn.ble_addr_type = p_lcb->ble_addr_type;
|
|
|
|
#endif
|
|
|
|
cb_data.conn.conn_handle = p_lcb->handle;
|
2019-05-23 14:44:16 +08:00
|
|
|
}else {
|
|
|
|
APPL_TRACE_WARNING("%s not found connection parameters of the device ", __func__);
|
|
|
|
}
|
|
|
|
}
|
2016-09-26 21:37:39 +08:00
|
|
|
cb_data.conn.conn_id = conn_id;
|
|
|
|
cb_data.conn.server_if = gatt_if;
|
|
|
|
cb_data.conn.reason = reason;
|
|
|
|
cb_data.conn.transport = transport;
|
|
|
|
memcpy(cb_data.conn.remote_bda, bda, BD_ADDR_LEN);
|
|
|
|
(*p_reg->p_cback)(evt, &cb_data);
|
2016-11-25 02:10:15 +08:00
|
|
|
} else {
|
|
|
|
APPL_TRACE_ERROR("bta_gatts_conn_cback server_if=%d not found", gatt_if);
|
2016-09-26 21:37:39 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*******************************************************************************
|
|
|
|
**
|
|
|
|
** Function bta_gatts_cong_cback
|
|
|
|
**
|
|
|
|
** Description congestion callback.
|
|
|
|
**
|
|
|
|
** Returns none.
|
|
|
|
**
|
|
|
|
*******************************************************************************/
|
|
|
|
static void bta_gatts_cong_cback (UINT16 conn_id, BOOLEAN congested)
|
|
|
|
{
|
|
|
|
tBTA_GATTS cb_data;
|
2020-05-13 20:30:35 +08:00
|
|
|
cb_data.congest.conn_id = conn_id;
|
|
|
|
cb_data.congest.congested = congested;
|
|
|
|
btc_congest_callback(&cb_data);
|
2016-09-26 21:37:39 +08:00
|
|
|
}
|
2017-04-20 07:18:05 -04:00
|
|
|
#endif /* GATTS_INCLUDED */
|