| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173 |
- /******************************************************************************
- *
- * Copyright (C) 1999-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 main GATT client functions
- *
- ******************************************************************************/
- #include "common/bt_target.h"
- #if BLE_INCLUDED == TRUE && GATTC_INCLUDED == TRUE
- #include <string.h>
- #include "osi/allocator.h"
- #include "gatt_int.h"
- #include "l2c_int.h"
- #define GATT_WRITE_LONG_HDR_SIZE 5 /* 1 opcode + 2 handle + 2 offset */
- #define GATT_READ_CHAR_VALUE_HDL (GATT_READ_CHAR_VALUE | 0x80)
- #define GATT_READ_INC_SRV_UUID128 (GATT_DISC_INC_SRVC | 0x90)
- #define GATT_PREP_WRITE_RSP_MIN_LEN 4
- #define GATT_NOTIFICATION_MIN_LEN 2
- #define GATT_WRITE_RSP_MIN_LEN 2
- #define GATT_INFO_RSP_MIN_LEN 1
- #define GATT_MTU_RSP_MIN_LEN 2
- #define GATT_READ_BY_TYPE_RSP_MIN_LEN 1
- /********************************************************************************
- ** G L O B A L G A T T D A T A *
- *********************************************************************************/
- void gatt_send_prepare_write(tGATT_TCB *p_tcb, tGATT_CLCB *p_clcb);
- static const UINT8 disc_type_to_att_opcode[GATT_DISC_MAX] = {
- 0,
- GATT_REQ_READ_BY_GRP_TYPE, /* GATT_DISC_SRVC_ALL = 1, */
- GATT_REQ_FIND_TYPE_VALUE, /* GATT_DISC_SRVC_BY_UUID, */
- GATT_REQ_READ_BY_TYPE, /* GATT_DISC_INC_SRVC, */
- GATT_REQ_READ_BY_TYPE, /* GATT_DISC_CHAR, */
- GATT_REQ_FIND_INFO /* GATT_DISC_CHAR_DSCPT, */
- };
- static const UINT16 disc_type_to_uuid[GATT_DISC_MAX] = {
- 0, /* reserved */
- GATT_UUID_PRI_SERVICE, /* <service> DISC_SRVC_ALL */
- GATT_UUID_PRI_SERVICE, /* <service> for DISC_SERVC_BY_UUID */
- GATT_UUID_INCLUDE_SERVICE, /* <include_service> for DISC_INC_SRVC */
- GATT_UUID_CHAR_DECLARE, /* <characteristic> for DISC_CHAR */
- 0 /* no type filtering for DISC_CHAR_DSCPT */
- };
- /*******************************************************************************
- **
- ** Function gatt_act_discovery
- **
- ** Description GATT discovery operation.
- **
- ** Returns void.
- **
- *******************************************************************************/
- void gatt_act_discovery(tGATT_CLCB *p_clcb)
- {
- UINT8 op_code = disc_type_to_att_opcode[p_clcb->op_subtype];
- tGATT_CL_MSG cl_req;
- tGATT_STATUS st;
- if (p_clcb->s_handle <= p_clcb->e_handle && p_clcb->s_handle != 0) {
- memset(&cl_req, 0, sizeof(tGATT_CL_MSG));
- cl_req.browse.s_handle = p_clcb->s_handle;
- cl_req.browse.e_handle = p_clcb->e_handle;
- if (disc_type_to_uuid[p_clcb->op_subtype] != 0) {
- cl_req.browse.uuid.len = 2;
- cl_req.browse.uuid.uu.uuid16 = disc_type_to_uuid[p_clcb->op_subtype];
- }
- if (p_clcb->op_subtype == GATT_DISC_SRVC_BY_UUID) { /* fill in the FindByTypeValue request info*/
- cl_req.find_type_value.uuid.len = 2;
- cl_req.find_type_value.uuid.uu.uuid16 = disc_type_to_uuid[p_clcb->op_subtype];
- cl_req.find_type_value.s_handle = p_clcb->s_handle;
- cl_req.find_type_value.e_handle = p_clcb->e_handle;
- cl_req.find_type_value.value_len = p_clcb->uuid.len;
- /* if service type is 32 bits UUID, convert it now */
- if (p_clcb->uuid.len == LEN_UUID_32) {
- cl_req.find_type_value.value_len = LEN_UUID_128;
- gatt_convert_uuid32_to_uuid128(cl_req.find_type_value.value, p_clcb->uuid.uu.uuid32);
- } else {
- memcpy (cl_req.find_type_value.value, &p_clcb->uuid.uu, p_clcb->uuid.len);
- }
- }
- st = attp_send_cl_msg(p_clcb->p_tcb, p_clcb->clcb_idx, op_code, &cl_req);
- if (st != GATT_SUCCESS && st != GATT_CMD_STARTED) {
- gatt_end_operation(p_clcb, GATT_ERROR, NULL);
- }
- } else { /* end of handle range */
- gatt_end_operation(p_clcb, GATT_SUCCESS, NULL);
- }
- }
- /*******************************************************************************
- **
- ** Function gatt_act_read
- **
- ** Description GATT read operation.
- **
- ** Returns void.
- **
- *******************************************************************************/
- void gatt_act_read (tGATT_CLCB *p_clcb, UINT16 offset)
- {
- tGATT_TCB *p_tcb = p_clcb->p_tcb;
- UINT8 rt = GATT_INTERNAL_ERROR;
- tGATT_CL_MSG msg;
- UINT8 op_code = 0;
- memset (&msg, 0, sizeof(tGATT_CL_MSG));
- switch (p_clcb->op_subtype) {
- case GATT_READ_CHAR_VALUE:
- case GATT_READ_BY_TYPE:
- op_code = GATT_REQ_READ_BY_TYPE;
- msg.browse.s_handle = p_clcb->s_handle;
- msg.browse.e_handle = p_clcb->e_handle;
- if (p_clcb->op_subtype == GATT_READ_BY_TYPE) {
- memcpy(&msg.browse.uuid, &p_clcb->uuid, sizeof(tBT_UUID));
- } else {
- msg.browse.uuid.len = LEN_UUID_16;
- msg.browse.uuid.uu.uuid16 = GATT_UUID_CHAR_DECLARE;
- }
- break;
- case GATT_READ_CHAR_VALUE_HDL:
- case GATT_READ_BY_HANDLE:
- if (!p_clcb->counter) {
- op_code = GATT_REQ_READ;
- msg.handle = p_clcb->s_handle;
- } else {
- if (!p_clcb->first_read_blob_after_read) {
- p_clcb->first_read_blob_after_read = TRUE;
- } else {
- p_clcb->first_read_blob_after_read = FALSE;
- }
- GATT_TRACE_DEBUG("gatt_act_read first_read_blob_after_read=%d",
- p_clcb->first_read_blob_after_read);
- op_code = GATT_REQ_READ_BLOB;
- msg.read_blob.offset = offset;
- msg.read_blob.handle = p_clcb->s_handle;
- }
- p_clcb->op_subtype &= ~ 0x80;
- break;
- case GATT_READ_PARTIAL:
- op_code = GATT_REQ_READ_BLOB;
- msg.read_blob.handle = p_clcb->s_handle;
- msg.read_blob.offset = offset;
- break;
- case GATT_READ_MULTIPLE:
- op_code = GATT_REQ_READ_MULTI;
- memcpy (&msg.read_multi, p_clcb->p_attr_buf, sizeof(tGATT_READ_MULTI));
- break;
- case GATT_READ_INC_SRV_UUID128:
- op_code = GATT_REQ_READ;
- msg.handle = p_clcb->s_handle;
- p_clcb->op_subtype &= ~ 0x90;
- break;
- default:
- GATT_TRACE_ERROR("Unknown read type: %d", p_clcb->op_subtype);
- break;
- }
- if (op_code != 0) {
- rt = attp_send_cl_msg(p_tcb, p_clcb->clcb_idx, op_code, &msg);
- }
- if ( op_code == 0 || (rt != GATT_SUCCESS && rt != GATT_CMD_STARTED)) {
- gatt_end_operation(p_clcb, rt, NULL);
- }
- }
- /*******************************************************************************
- **
- ** Function gatt_act_write
- **
- ** Description GATT write operation.
- **
- ** Returns void.
- **
- *******************************************************************************/
- void gatt_act_write (tGATT_CLCB *p_clcb, UINT8 sec_act)
- {
- tGATT_TCB *p_tcb = p_clcb->p_tcb;
- UINT8 rt = GATT_SUCCESS, op_code = 0;
- tGATT_VALUE *p_attr = (tGATT_VALUE *)p_clcb->p_attr_buf;
- if (p_attr) {
- switch (p_clcb->op_subtype) {
- case GATT_WRITE_NO_RSP:
- l2ble_update_att_acl_pkt_num(L2CA_DECREASE_BTU_NUM, NULL);
- p_clcb->s_handle = p_attr->handle;
- op_code = (sec_act == GATT_SEC_SIGN_DATA) ? GATT_SIGN_CMD_WRITE : GATT_CMD_WRITE;
- rt = gatt_send_write_msg(p_tcb,
- p_clcb->clcb_idx,
- op_code,
- p_attr->handle,
- p_attr->len,
- 0,
- p_attr->value);
- break;
- case GATT_WRITE:
- if (p_attr->len <= (p_tcb->payload_size - GATT_HDR_SIZE)) {
- p_clcb->s_handle = p_attr->handle;
- rt = gatt_send_write_msg(p_tcb,
- p_clcb->clcb_idx,
- GATT_REQ_WRITE,
- p_attr->handle,
- p_attr->len,
- 0,
- p_attr->value);
- } else { /* prepare write for long attribute */
- gatt_send_prepare_write(p_tcb, p_clcb);
- }
- break;
- case GATT_WRITE_PREPARE:
- gatt_send_prepare_write(p_tcb, p_clcb);
- break;
- default:
- rt = GATT_INTERNAL_ERROR;
- GATT_TRACE_ERROR("Unknown write type: %d", p_clcb->op_subtype);
- break;
- }
- } else {
- rt = GATT_INTERNAL_ERROR;
- }
- if ((rt != GATT_SUCCESS && rt != GATT_CMD_STARTED && rt != GATT_CONGESTED)
- || (rt != GATT_CMD_STARTED && p_clcb->op_subtype == GATT_WRITE_NO_RSP)) {
- if (rt != GATT_SUCCESS) {
- GATT_TRACE_DEBUG("gatt_act_write() failed op_code=0x%x rt=%d", op_code, rt);
- }
- gatt_end_operation(p_clcb, rt, NULL);
- }
- }
- /*******************************************************************************
- **
- ** Function gatt_send_queue_write_cancel
- **
- ** Description send queue write cancel
- **
- ** Returns void.
- **
- *******************************************************************************/
- void gatt_send_queue_write_cancel (tGATT_TCB *p_tcb, tGATT_CLCB *p_clcb, tGATT_EXEC_FLAG flag)
- {
- UINT8 rt ;
- GATT_TRACE_DEBUG("gatt_send_queue_write_cancel ");
- rt = attp_send_cl_msg(p_tcb, p_clcb->clcb_idx, GATT_REQ_EXEC_WRITE, (tGATT_CL_MSG *)&flag);
- if (rt != GATT_SUCCESS) {
- gatt_end_operation(p_clcb, rt, NULL);
- }
- }
- /*******************************************************************************
- **
- ** Function gatt_check_write_long_terminate
- **
- ** Description To terminate write long or not.
- **
- ** Returns TRUE: write long is terminated; FALSE keep sending.
- **
- *******************************************************************************/
- BOOLEAN gatt_check_write_long_terminate(tGATT_TCB *p_tcb, tGATT_CLCB *p_clcb, tGATT_VALUE *p_rsp_value)
- {
- tGATT_VALUE *p_attr = (tGATT_VALUE *)p_clcb->p_attr_buf;
- BOOLEAN exec = FALSE;
- tGATT_EXEC_FLAG flag = GATT_PREP_WRITE_EXEC;
- GATT_TRACE_DEBUG("gatt_check_write_long_terminate ");
- /* check the first write response status */
- if (p_rsp_value != NULL) {
- if (p_rsp_value->handle != p_attr->handle ||
- p_rsp_value->len != p_clcb->counter ||
- memcmp(p_rsp_value->value, p_attr->value + p_attr->offset, p_rsp_value->len)) {
- /* data does not match */
- p_clcb->status = GATT_ERROR;
- flag = GATT_PREP_WRITE_CANCEL;
- exec = TRUE;
- } else { /* response checking is good */
- p_clcb->status = GATT_SUCCESS;
- /* update write offset and check if end of attribute value */
- if ((p_attr->offset += p_rsp_value->len) >= p_attr->len) {
- exec = TRUE;
- }
- }
- }
- if (exec) {
- gatt_send_queue_write_cancel (p_tcb, p_clcb, flag);
- return TRUE;
- }
- return FALSE;
- }
- /*******************************************************************************
- **
- ** Function gatt_send_prepare_write
- **
- ** Description Send prepare write.
- **
- ** Returns void.
- **
- *******************************************************************************/
- void gatt_send_prepare_write(tGATT_TCB *p_tcb, tGATT_CLCB *p_clcb)
- {
- tGATT_VALUE *p_attr = (tGATT_VALUE *)p_clcb->p_attr_buf;
- UINT16 to_send, offset;
- UINT8 rt = GATT_SUCCESS;
- UINT8 type = p_clcb->op_subtype;
- GATT_TRACE_DEBUG("gatt_send_prepare_write type=0x%x", type );
- to_send = p_attr->len - p_attr->offset;
- if (to_send > (p_tcb->payload_size - GATT_WRITE_LONG_HDR_SIZE)) { /* 2 = UINT16 offset bytes */
- to_send = p_tcb->payload_size - GATT_WRITE_LONG_HDR_SIZE;
- }
- p_clcb->s_handle = p_attr->handle;
- offset = p_attr->offset;
- if (type == GATT_WRITE_PREPARE) {
- offset += p_clcb->start_offset;
- }
- GATT_TRACE_DEBUG("offset =0x%x len=%d", offset, to_send );
- rt = gatt_send_write_msg(p_tcb,
- p_clcb->clcb_idx,
- GATT_REQ_PREPARE_WRITE,
- p_attr->handle,
- to_send, /* length */
- offset, /* used as offset */
- p_attr->value + p_attr->offset); /* data */
- /* remember the write long attribute length */
- p_clcb->counter = to_send;
- if (rt != GATT_SUCCESS && rt != GATT_CMD_STARTED) {
- gatt_end_operation(p_clcb, rt, NULL);
- }
- }
- /*******************************************************************************
- **
- ** Function gatt_process_find_type_value_rsp
- **
- ** Description This function is called to handle find by type value response.
- **
- **
- ** Returns void
- **
- *******************************************************************************/
- void gatt_process_find_type_value_rsp (tGATT_TCB *p_tcb, tGATT_CLCB *p_clcb, UINT16 len, UINT8 *p_data)
- {
- tGATT_DISC_RES result;
- UINT8 *p = p_data;
- UNUSED(p_tcb);
- GATT_TRACE_DEBUG("gatt_process_find_type_value_rsp ");
- /* unexpected response */
- if (p_clcb->operation != GATTC_OPTYPE_DISCOVERY || p_clcb->op_subtype != GATT_DISC_SRVC_BY_UUID) {
- return;
- }
- memset (&result, 0, sizeof(tGATT_DISC_RES));
- result.type.len = 2;
- result.type.uu.uuid16 = GATT_UUID_PRI_SERVICE;
- /* returns a series of handle ranges */
- while (len >= 4) {
- STREAM_TO_UINT16 (result.handle, p);
- STREAM_TO_UINT16 (result.value.group_value.e_handle, p);
- memcpy (&result.value.group_value.service_type, &p_clcb->uuid, sizeof(tBT_UUID));
- len -= 4;
- if (p_clcb->p_reg->app_cb.p_disc_res_cb) {
- (*p_clcb->p_reg->app_cb.p_disc_res_cb)(p_clcb->conn_id, p_clcb->op_subtype, &result);
- }
- }
- /* last handle + 1 */
- p_clcb->s_handle = (result.value.group_value.e_handle == 0) ? 0 : (result.value.group_value.e_handle + 1);
- /* initiate another request */
- gatt_act_discovery(p_clcb) ;
- }
- /*******************************************************************************
- **
- ** Function gatt_process_read_info_rsp
- **
- ** Description This function is called to handle the read information
- ** response.
- **
- **
- ** Returns void
- **
- *******************************************************************************/
- void gatt_process_read_info_rsp(tGATT_TCB *p_tcb, tGATT_CLCB *p_clcb, UINT8 op_code,
- UINT16 len, UINT8 *p_data)
- {
- tGATT_DISC_RES result;
- UINT8 *p = p_data, uuid_len = 0, type;
- UNUSED(p_tcb);
- UNUSED(op_code);
- if (len < GATT_INFO_RSP_MIN_LEN) {
- GATT_TRACE_ERROR("invalid Info Response PDU received, discard.");
- gatt_end_operation(p_clcb, GATT_INVALID_PDU, NULL);
- return;
- }
- /* unexpected response */
- if (p_clcb->operation != GATTC_OPTYPE_DISCOVERY || p_clcb->op_subtype != GATT_DISC_CHAR_DSCPT) {
- return;
- }
- STREAM_TO_UINT8(type, p);
- len -= 1;
- if (type == GATT_INFO_TYPE_PAIR_16) {
- uuid_len = LEN_UUID_16;
- } else if (type == GATT_INFO_TYPE_PAIR_128) {
- uuid_len = LEN_UUID_128;
- }
- while (len >= uuid_len + 2) {
- STREAM_TO_UINT16 (result.handle, p);
- if (uuid_len > 0) {
- if (!gatt_parse_uuid_from_cmd(&result.type, uuid_len, &p)) {
- break;
- }
- } else {
- memcpy (&result.type, &p_clcb->uuid, sizeof(tBT_UUID));
- }
- len -= (uuid_len + 2);
- if (p_clcb->p_reg->app_cb.p_disc_res_cb) {
- (*p_clcb->p_reg->app_cb.p_disc_res_cb)(p_clcb->conn_id, p_clcb->op_subtype, &result);
- }
- }
- p_clcb->s_handle = (result.handle == 0) ? 0 : (result.handle + 1);
- /* initiate another request */
- gatt_act_discovery(p_clcb) ;
- }
- /*******************************************************************************
- **
- ** Function gatt_proc_disc_error_rsp
- **
- ** Description This function process the read by type response and send another
- ** request if needed.
- **
- ** Returns void.
- **
- *******************************************************************************/
- void gatt_proc_disc_error_rsp(tGATT_TCB *p_tcb, tGATT_CLCB *p_clcb, UINT8 opcode,
- UINT16 handle, UINT8 reason)
- {
- tGATT_STATUS status = (tGATT_STATUS) reason;
- UNUSED(p_tcb);
- UNUSED(handle);
- GATT_TRACE_DEBUG("gatt_proc_disc_error_rsp reason: %02x cmd_code %04x", reason, opcode);
- switch (opcode) {
- case GATT_REQ_READ_BY_GRP_TYPE:
- case GATT_REQ_FIND_TYPE_VALUE:
- case GATT_REQ_READ_BY_TYPE:
- case GATT_REQ_FIND_INFO:
- if (reason == GATT_NOT_FOUND) {
- status = GATT_SUCCESS;
- GATT_TRACE_DEBUG("Discovery completed");
- }
- break;
- default:
- GATT_TRACE_ERROR("Incorrect discovery opcode %04x", opcode);
- break;
- }
- gatt_end_operation(p_clcb, status, NULL);
- }
- /*******************************************************************************
- **
- ** Function gatt_process_error_rsp
- **
- ** Description This function is called to handle the error response
- **
- **
- ** Returns void
- **
- *******************************************************************************/
- void gatt_process_error_rsp(tGATT_TCB *p_tcb, tGATT_CLCB *p_clcb, UINT8 op_code,
- UINT16 len, UINT8 *p_data)
- {
- UINT8 opcode, reason, * p = p_data;
- UINT16 handle;
- tGATT_VALUE *p_attr = (tGATT_VALUE *)p_clcb->p_attr_buf;
- UNUSED(op_code);
- UNUSED(len);
- GATT_TRACE_DEBUG("gatt_process_error_rsp ");
- STREAM_TO_UINT8(opcode, p);
- STREAM_TO_UINT16(handle, p);
- STREAM_TO_UINT8(reason, p);
- if (p_clcb->operation == GATTC_OPTYPE_DISCOVERY) {
- gatt_proc_disc_error_rsp(p_tcb, p_clcb, opcode, handle, reason);
- } else {
- if ( (p_clcb->operation == GATTC_OPTYPE_WRITE) &&
- (p_clcb->op_subtype == GATT_WRITE) &&
- (opcode == GATT_REQ_PREPARE_WRITE) &&
- (p_attr) &&
- (handle == p_attr->handle) ) {
- if (reason == GATT_SUCCESS){
- reason = GATT_ERROR;
- }
- p_clcb->status = reason;
- gatt_send_queue_write_cancel(p_tcb, p_clcb, GATT_PREP_WRITE_CANCEL);
- } else if ((p_clcb->operation == GATTC_OPTYPE_READ) &&
- ((p_clcb->op_subtype == GATT_READ_CHAR_VALUE_HDL) ||
- (p_clcb->op_subtype == GATT_READ_BY_HANDLE)) &&
- (opcode == GATT_REQ_READ_BLOB) &&
- p_clcb->first_read_blob_after_read &&
- (reason == GATT_NOT_LONG)) {
- gatt_end_operation(p_clcb, GATT_SUCCESS, (void *)p_clcb->p_attr_buf);
- } else {
- gatt_end_operation(p_clcb, reason, NULL);
- }
- }
- }
- /*******************************************************************************
- **
- ** Function gatt_process_prep_write_rsp
- **
- ** Description This function is called to handle the read response
- **
- **
- ** Returns void
- **
- *******************************************************************************/
- void gatt_process_prep_write_rsp (tGATT_TCB *p_tcb, tGATT_CLCB *p_clcb, UINT8 op_code,
- UINT16 len, UINT8 *p_data)
- {
- tGATT_VALUE value = {0};
- UINT8 *p = p_data;
- GATT_TRACE_DEBUG("value resp op_code = %s len = %d", gatt_dbg_op_name(op_code), len);
- if (len < GATT_PREP_WRITE_RSP_MIN_LEN) {
- GATT_TRACE_ERROR("illegal prepare write response length, discard");
- gatt_end_operation(p_clcb, GATT_INVALID_PDU, &value);
- return;
- }
- STREAM_TO_UINT16 (value.handle, p);
- STREAM_TO_UINT16 (value.offset, p);
- value.len = len - 4;
- memcpy (value.value, p, value.len);
- if (p_clcb->op_subtype == GATT_WRITE_PREPARE) {
- p_clcb->status = GATT_SUCCESS;
- /* application should verify handle offset
- and value are matched or not */
- gatt_end_operation(p_clcb, p_clcb->status, &value);
- } else if (p_clcb->op_subtype == GATT_WRITE ) {
- if (!gatt_check_write_long_terminate(p_tcb, p_clcb, &value)) {
- gatt_send_prepare_write(p_tcb, p_clcb);
- }
- }
- }
- /*******************************************************************************
- **
- ** Function gatt_process_notification
- **
- ** Description This function is called to handle the handle value indication
- ** or handle value notification.
- **
- **
- ** Returns void
- **
- *******************************************************************************/
- void gatt_process_notification(tGATT_TCB *p_tcb, UINT8 op_code,
- UINT16 len, UINT8 *p_data)
- {
- tGATT_VALUE value = {0};
- tGATT_REG *p_reg;
- UINT16 conn_id;
- tGATT_STATUS encrypt_status;
- UINT8 *p = p_data, i,
- event = (op_code == GATT_HANDLE_VALUE_NOTIF) ? GATTC_OPTYPE_NOTIFICATION : GATTC_OPTYPE_INDICATION;
- GATT_TRACE_DEBUG("gatt_process_notification ");
- if (len < GATT_NOTIFICATION_MIN_LEN) {
- GATT_TRACE_ERROR("illegal notification PDU length, discard");
- return;
- }
- STREAM_TO_UINT16 (value.handle, p);
- value.len = len - 2;
- memcpy (value.value, p, value.len);
- if (!GATT_HANDLE_IS_VALID(value.handle)) {
- /* illegal handle, send ack now */
- if (op_code == GATT_HANDLE_VALUE_IND) {
- attp_send_cl_msg(p_tcb, 0, GATT_HANDLE_VALUE_CONF, NULL);
- }
- return;
- }
- if (event == GATTC_OPTYPE_INDICATION) {
- if (p_tcb->ind_count) {
- /* this is an error case that receiving an indication but we
- still has an indication not being acked yet.
- For now, just log the error reset the counter.
- Later we need to disconnect the link unconditionally.
- */
- GATT_TRACE_ERROR("gatt_process_notification rcv Ind. but ind_count=%d (will reset ind_count)", p_tcb->ind_count);
- }
- p_tcb->ind_count = 0;
- }
- /* should notify all registered client with the handle value notificaion/indication
- Note: need to do the indication count and start timer first then do callback
- */
- for (i = 0, p_reg = gatt_cb.cl_rcb; i < GATT_MAX_APPS; i++, p_reg++) {
- if (p_reg->in_use && p_reg->app_cb.p_cmpl_cb && (event == GATTC_OPTYPE_INDICATION)) {
- p_tcb->ind_count++;
- }
- }
- if (event == GATTC_OPTYPE_INDICATION) {
- /* start a timer for app confirmation */
- if (p_tcb->ind_count > 0) {
- gatt_start_ind_ack_timer(p_tcb);
- } else { /* no app to indicate, or invalid handle */
- attp_send_cl_msg(p_tcb, 0, GATT_HANDLE_VALUE_CONF, NULL);
- }
- }
- encrypt_status = gatt_get_link_encrypt_status(p_tcb);
- for (i = 0, p_reg = gatt_cb.cl_rcb; i < GATT_MAX_APPS; i++, p_reg++) {
- if (p_reg->in_use && p_reg->app_cb.p_cmpl_cb) {
- conn_id = GATT_CREATE_CONN_ID(p_tcb->tcb_idx, p_reg->gatt_if);
- (*p_reg->app_cb.p_cmpl_cb) (conn_id, event, encrypt_status, (tGATT_CL_COMPLETE *)&value);
- }
- }
- }
- /*******************************************************************************
- **
- ** Function gatt_process_read_by_type_rsp
- **
- ** Description This function is called to handle the read by type response.
- ** read by type can be used for discovery, or read by type or
- ** read characteristic value.
- **
- ** Returns void
- **
- *******************************************************************************/
- void gatt_process_read_by_type_rsp (tGATT_TCB *p_tcb, tGATT_CLCB *p_clcb, UINT8 op_code,
- UINT16 len, UINT8 *p_data)
- {
- tGATT_DISC_RES result;
- tGATT_DISC_VALUE record_value;
- UINT8 *p = p_data, value_len, handle_len = 2;
- UINT16 handle = 0;
- /* discovery procedure and no callback function registered */
- if (((!p_clcb->p_reg) || (!p_clcb->p_reg->app_cb.p_disc_res_cb)) && (p_clcb->operation == GATTC_OPTYPE_DISCOVERY)) {
- return;
- }
- if (len < GATT_READ_BY_TYPE_RSP_MIN_LEN) {
- GATT_TRACE_ERROR("Illegal ReadByType/ReadByGroupType Response length, discard");
- gatt_end_operation(p_clcb, GATT_INVALID_PDU, NULL);
- return;
- }
- STREAM_TO_UINT8(value_len, p);
- if ((value_len > (p_tcb->payload_size - 2)) || (value_len > (len - 1)) ) {
- /* this is an error case that server's response containing a value length which is larger than MTU-2
- or value_len > message total length -1 */
- GATT_TRACE_ERROR("gatt_process_read_by_type_rsp: Discard response op_code=%d vale_len=%d > (MTU-2=%d or msg_len-1=%d)",
- op_code, value_len, (p_tcb->payload_size - 2), (len - 1));
- gatt_end_operation(p_clcb, GATT_ERROR, NULL);
- return;
- }
- if (op_code == GATT_RSP_READ_BY_GRP_TYPE) {
- handle_len = 4;
- }
- value_len -= handle_len; /* substract the handle pairs bytes */
- len -= 1;
- while (len >= (handle_len + value_len)) {
- STREAM_TO_UINT16(handle, p);
- if (!GATT_HANDLE_IS_VALID(handle)) {
- gatt_end_operation(p_clcb, GATT_INVALID_HANDLE, NULL);
- return;
- }
- memset(&result, 0, sizeof(tGATT_DISC_RES));
- memset(&record_value, 0, sizeof(tGATT_DISC_VALUE));
- result.handle = handle;
- result.type.len = 2;
- result.type.uu.uuid16 = disc_type_to_uuid[p_clcb->op_subtype];
- /* discover all services */
- if (p_clcb->operation == GATTC_OPTYPE_DISCOVERY &&
- p_clcb->op_subtype == GATT_DISC_SRVC_ALL &&
- op_code == GATT_RSP_READ_BY_GRP_TYPE) {
- STREAM_TO_UINT16(handle, p);
- if (!GATT_HANDLE_IS_VALID(handle)) {
- gatt_end_operation(p_clcb, GATT_INVALID_HANDLE, NULL);
- return;
- } else {
- record_value.group_value.e_handle = handle;
- if (!gatt_parse_uuid_from_cmd(&record_value.group_value.service_type, value_len, &p)) {
- GATT_TRACE_ERROR("discover all service response parsing failure");
- break;
- }
- }
- }
- /* discover included service */
- else if (p_clcb->operation == GATTC_OPTYPE_DISCOVERY && p_clcb->op_subtype == GATT_DISC_INC_SRVC) {
- STREAM_TO_UINT16(record_value.incl_service.s_handle, p);
- STREAM_TO_UINT16(record_value.incl_service.e_handle, p);
- if (!GATT_HANDLE_IS_VALID(record_value.incl_service.s_handle) ||
- !GATT_HANDLE_IS_VALID(record_value.incl_service.e_handle)) {
- gatt_end_operation(p_clcb, GATT_INVALID_HANDLE, NULL);
- return;
- }
- if (value_len == 6) {
- STREAM_TO_UINT16(record_value.incl_service.service_type.uu.uuid16, p);
- record_value.incl_service.service_type.len = LEN_UUID_16;
- } else if (value_len == 4) {
- p_clcb->s_handle = record_value.incl_service.s_handle;
- p_clcb->read_uuid128.wait_for_read_rsp = TRUE;
- p_clcb->read_uuid128.next_disc_start_hdl = handle + 1;
- memcpy(&p_clcb->read_uuid128.result, &result, sizeof(result));
- memcpy(&p_clcb->read_uuid128.result.value, &record_value, sizeof (result.value));
- p_clcb->op_subtype |= 0x90;
- gatt_act_read(p_clcb, 0);
- return;
- } else {
- GATT_TRACE_ERROR("gatt_process_read_by_type_rsp INCL_SRVC failed with invalid data value_len=%d", value_len);
- gatt_end_operation(p_clcb, GATT_INVALID_PDU, (void *)p);
- return;
- }
- }
- /* read by type */
- else if (p_clcb->operation == GATTC_OPTYPE_READ && p_clcb->op_subtype == GATT_READ_BY_TYPE) {
- p_clcb->counter = len - 2;
- p_clcb->s_handle = handle;
- if ( p_clcb->counter == (p_clcb->p_tcb->payload_size - 4)) {
- p_clcb->op_subtype = GATT_READ_BY_HANDLE;
- if (!p_clcb->p_attr_buf) {
- p_clcb->p_attr_buf = (UINT8 *)osi_malloc(GATT_MAX_ATTR_LEN);
- }
- if (p_clcb->p_attr_buf && p_clcb->counter <= GATT_MAX_ATTR_LEN) {
- memcpy(p_clcb->p_attr_buf, p, p_clcb->counter);
- gatt_act_read(p_clcb, p_clcb->counter);
- } else {
- gatt_end_operation(p_clcb, GATT_INTERNAL_ERROR, (void *)p);
- }
- } else {
- gatt_end_operation(p_clcb, GATT_SUCCESS, (void *)p);
- }
- return;
- } else { /* discover characterisitic */
- STREAM_TO_UINT8 (record_value.dclr_value.char_prop, p);
- STREAM_TO_UINT16(record_value.dclr_value.val_handle, p);
- if (!GATT_HANDLE_IS_VALID(record_value.dclr_value.val_handle)) {
- gatt_end_operation(p_clcb, GATT_INVALID_HANDLE, NULL);
- return;
- }
- if (!gatt_parse_uuid_from_cmd(&record_value.dclr_value.char_uuid, (UINT16)(value_len - 3), &p)) {
- gatt_end_operation(p_clcb, GATT_SUCCESS, NULL);
- /* invalid format, and skip the result */
- return;
- }
- /* UUID not matching */
- if (!gatt_uuid_compare(record_value.dclr_value.char_uuid, p_clcb->uuid)) {
- len -= (value_len + 2);
- continue; /* skip the result, and look for next one */
- } else if (p_clcb->operation == GATTC_OPTYPE_READ)
- /* UUID match for read characteristic value */
- {
- /* only read the first matching UUID characteristic value, and
- discard the rest results */
- p_clcb->s_handle = record_value.dclr_value.val_handle;
- p_clcb->op_subtype |= 0x80;
- gatt_act_read(p_clcb, 0);
- return;
- }
- }
- len -= (value_len + handle_len);
- /* result is (handle, 16bits UUID) pairs */
- memcpy (&result.value, &record_value, sizeof (result.value));
- /* send callback if is discover procedure */
- if (p_clcb->operation == GATTC_OPTYPE_DISCOVERY && p_clcb->p_reg->app_cb.p_disc_res_cb) {
- (*p_clcb->p_reg->app_cb.p_disc_res_cb)(p_clcb->conn_id, p_clcb->op_subtype, &result);
- }
- }
- p_clcb->s_handle = (handle == 0) ? 0 : (handle + 1);
- if (p_clcb->operation == GATTC_OPTYPE_DISCOVERY) {
- /* initiate another request */
- gatt_act_discovery(p_clcb) ;
- } else { /* read characteristic value */
- gatt_act_read(p_clcb, 0);
- }
- }
- /*******************************************************************************
- **
- ** Function gatt_process_read_rsp
- **
- ** Description This function is called to handle the read BLOB response
- **
- **
- ** Returns void
- **
- *******************************************************************************/
- void gatt_process_read_rsp(tGATT_TCB *p_tcb, tGATT_CLCB *p_clcb, UINT8 op_code,
- UINT16 len, UINT8 *p_data)
- {
- UINT16 offset = p_clcb->counter;
- UINT8 *p = p_data;
- UNUSED(op_code);
- if (p_clcb->operation == GATTC_OPTYPE_READ) {
- if (p_clcb->op_subtype != GATT_READ_BY_HANDLE) {
- p_clcb->counter = len;
- gatt_end_operation(p_clcb, GATT_SUCCESS, (void *)p);
- } else {
- /* allocate GKI buffer holding up long attribute value */
- if (!p_clcb->p_attr_buf) {
- p_clcb->p_attr_buf = (UINT8 *)osi_malloc(GATT_MAX_ATTR_LEN);
- }
- /* copy attrobute value into cb buffer */
- if (p_clcb->p_attr_buf && offset < GATT_MAX_ATTR_LEN) {
- if ((len + offset) > GATT_MAX_ATTR_LEN) {
- len = GATT_MAX_ATTR_LEN - offset;
- }
- p_clcb->counter += len;
- memcpy(p_clcb->p_attr_buf + offset, p, len);
- /* send next request if needed */
- if (len == (p_tcb->payload_size - 1) && /* full packet for read or read blob rsp */
- len + offset < GATT_MAX_ATTR_LEN) {
- GATT_TRACE_DEBUG("full pkt issue read blob for remianing bytes old offset=%d len=%d new offset=%d",
- offset, len, p_clcb->counter);
- gatt_act_read(p_clcb, p_clcb->counter);
- } else { /* end of request, send callback */
- gatt_end_operation(p_clcb, GATT_SUCCESS, (void *)p_clcb->p_attr_buf);
- }
- } else { /* exception, should not happen */
- GATT_TRACE_ERROR("attr offset = %d p_attr_buf = %p ", offset, p_clcb->p_attr_buf);
- gatt_end_operation(p_clcb, GATT_NO_RESOURCES, (void *)p_clcb->p_attr_buf);
- }
- }
- } else {
- if (p_clcb->operation == GATTC_OPTYPE_DISCOVERY &&
- p_clcb->op_subtype == GATT_DISC_INC_SRVC &&
- p_clcb->read_uuid128.wait_for_read_rsp ) {
- p_clcb->s_handle = p_clcb->read_uuid128.next_disc_start_hdl;
- p_clcb->read_uuid128.wait_for_read_rsp = FALSE;
- if (len == LEN_UUID_128) {
- memcpy(p_clcb->read_uuid128.result.value.incl_service.service_type.uu.uuid128, p, len);
- p_clcb->read_uuid128.result.value.incl_service.service_type.len = LEN_UUID_128;
- if ( p_clcb->p_reg->app_cb.p_disc_res_cb) {
- (*p_clcb->p_reg->app_cb.p_disc_res_cb)(p_clcb->conn_id, p_clcb->op_subtype, &p_clcb->read_uuid128.result);
- }
- gatt_act_discovery(p_clcb) ;
- } else {
- gatt_end_operation(p_clcb, GATT_INVALID_PDU, (void *)p);
- }
- }
- }
- }
- /*******************************************************************************
- **
- ** Function gatt_process_handle_rsp
- **
- ** Description This function is called to handle the write response
- **
- **
- ** Returns void
- **
- *******************************************************************************/
- void gatt_process_handle_rsp(tGATT_CLCB *p_clcb)
- {
- gatt_end_operation(p_clcb, GATT_SUCCESS, NULL);
- }
- /*******************************************************************************
- **
- ** Function gatt_process_mtu_rsp
- **
- ** Description This function is called to process the configure MTU response.
- **
- **
- ** Returns void
- **
- *******************************************************************************/
- void gatt_process_mtu_rsp(tGATT_TCB *p_tcb, tGATT_CLCB *p_clcb, UINT16 len, UINT8 *p_data)
- {
- UINT16 mtu;
- tGATT_STATUS status = GATT_SUCCESS;
- if (len < GATT_MTU_RSP_MIN_LEN) {
- GATT_TRACE_ERROR("invalid MTU response PDU received, discard.");
- status = GATT_INVALID_PDU;
- } else {
- STREAM_TO_UINT16(mtu, p_data);
- if (mtu < p_tcb->payload_size && mtu >= GATT_DEF_BLE_MTU_SIZE) {
- p_tcb->payload_size = mtu;
- }
- }
- /* host will set packet data length to 251 automatically if remote device support set packet data length,
- so l2cble_set_fixed_channel_tx_data_length() is not necessary.
- l2cble_set_fixed_channel_tx_data_length(p_tcb->peer_bda, L2CAP_ATT_CID, p_tcb->payload_size);
- */
- gatt_end_operation(p_clcb, status, NULL);
- }
- /*******************************************************************************
- **
- ** Function gatt_cmd_to_rsp_code
- **
- ** Description The function convert a ATT command op code into the corresponding
- ** response code assume no error occurs.
- **
- ** Returns response code.
- **
- *******************************************************************************/
- UINT8 gatt_cmd_to_rsp_code (UINT8 cmd_code)
- {
- UINT8 rsp_code = 0;
- if (cmd_code > 1 && cmd_code != GATT_CMD_WRITE) {
- rsp_code = cmd_code + 1;
- }
- return rsp_code;
- }
- /*******************************************************************************
- **
- ** Function gatt_cl_send_next_cmd_inq
- **
- ** Description Find next command in queue and sent to server
- **
- ** Returns TRUE if command sent, otherwise FALSE.
- **
- *******************************************************************************/
- BOOLEAN gatt_cl_send_next_cmd_inq(tGATT_TCB *p_tcb)
- {
- tGATT_CMD_Q *p_cmd = &p_tcb->cl_cmd_q[p_tcb->pending_cl_req];
- BOOLEAN sent = FALSE;
- UINT8 rsp_code;
- tGATT_CLCB *p_clcb = NULL;
- tGATT_STATUS att_ret = GATT_SUCCESS;
- while (!sent &&
- p_tcb->pending_cl_req != p_tcb->next_slot_inq &&
- p_cmd->to_send && p_cmd->p_cmd != NULL) {
- att_ret = attp_send_msg_to_l2cap(p_tcb, p_cmd->p_cmd);
- if (att_ret == GATT_SUCCESS || att_ret == GATT_CONGESTED) {
- sent = TRUE;
- p_cmd->to_send = FALSE;
- if(p_cmd->p_cmd) {
- osi_free(p_cmd->p_cmd);
- p_cmd->p_cmd = NULL;
- }
- /* dequeue the request if is write command or sign write */
- if (p_cmd->op_code != GATT_CMD_WRITE && p_cmd->op_code != GATT_SIGN_CMD_WRITE) {
- gatt_start_rsp_timer (p_cmd->clcb_idx);
- } else {
- p_clcb = gatt_cmd_dequeue(p_tcb, &rsp_code);
- /* if no ack needed, keep sending */
- if (att_ret == GATT_SUCCESS) {
- sent = FALSE;
- }
- p_cmd = &p_tcb->cl_cmd_q[p_tcb->pending_cl_req];
- /* send command complete callback here */
- gatt_end_operation(p_clcb, att_ret, NULL);
- }
- } else {
- GATT_TRACE_ERROR("gatt_cl_send_next_cmd_inq: L2CAP sent error");
- memset(p_cmd, 0, sizeof(tGATT_CMD_Q));
- p_tcb->pending_cl_req ++;
- p_cmd = &p_tcb->cl_cmd_q[p_tcb->pending_cl_req];
- }
- }
- return sent;
- }
- /*******************************************************************************
- **
- ** Function gatt_client_handle_server_rsp
- **
- ** Description This function is called to handle the server response to
- ** client.
- **
- **
- ** Returns void
- **
- *******************************************************************************/
- void gatt_client_handle_server_rsp (tGATT_TCB *p_tcb, UINT8 op_code,
- UINT16 len, UINT8 *p_data)
- {
- tGATT_CLCB *p_clcb = NULL;
- UINT8 rsp_code;
- if (op_code != GATT_HANDLE_VALUE_IND && op_code != GATT_HANDLE_VALUE_NOTIF) {
- p_clcb = gatt_cmd_dequeue(p_tcb, &rsp_code);
- rsp_code = gatt_cmd_to_rsp_code(rsp_code);
- if (p_clcb == NULL || (rsp_code != op_code && op_code != GATT_RSP_ERROR)) {
- GATT_TRACE_WARNING ("ATT - Ignore wrong response. Receives (%02x) \
- Request(%02x) Ignored", op_code, rsp_code);
- return;
- } else {
- btu_stop_timer (&p_clcb->rsp_timer_ent);
- p_clcb->retry_count = 0;
- }
- }
- /* the size of the message may not be bigger than the local max PDU size*/
- /* The message has to be smaller than the agreed MTU, len does not count op_code */
- if (len >= p_tcb->payload_size) {
- GATT_TRACE_ERROR("invalid response/indicate pkt size: %d, PDU size: %d", len + 1, p_tcb->payload_size);
- if (op_code != GATT_HANDLE_VALUE_NOTIF &&
- op_code != GATT_HANDLE_VALUE_IND) {
- gatt_end_operation(p_clcb, GATT_ERROR, NULL);
- }
- } else {
- switch (op_code) {
- case GATT_RSP_ERROR:
- gatt_process_error_rsp(p_tcb, p_clcb, op_code, len, p_data);
- break;
- case GATT_RSP_MTU: /* 2 bytes mtu */
- gatt_process_mtu_rsp(p_tcb, p_clcb, len , p_data);
- break;
- case GATT_RSP_FIND_INFO:
- gatt_process_read_info_rsp(p_tcb, p_clcb, op_code, len, p_data);
- break;
- case GATT_RSP_READ_BY_TYPE:
- case GATT_RSP_READ_BY_GRP_TYPE:
- gatt_process_read_by_type_rsp(p_tcb, p_clcb, op_code, len, p_data);
- break;
- case GATT_RSP_READ:
- case GATT_RSP_READ_BLOB:
- case GATT_RSP_READ_MULTI:
- gatt_process_read_rsp(p_tcb, p_clcb, op_code, len, p_data);
- break;
- case GATT_RSP_FIND_TYPE_VALUE: /* disc service with UUID */
- gatt_process_find_type_value_rsp(p_tcb, p_clcb, len, p_data);
- break;
- case GATT_RSP_WRITE:
- gatt_process_handle_rsp(p_clcb);
- break;
- case GATT_RSP_PREPARE_WRITE:
- gatt_process_prep_write_rsp(p_tcb, p_clcb, op_code, len, p_data);
- break;
- case GATT_RSP_EXEC_WRITE:
- gatt_end_operation(p_clcb, p_clcb->status, NULL);
- break;
- case GATT_HANDLE_VALUE_NOTIF:
- case GATT_HANDLE_VALUE_IND:
- gatt_process_notification(p_tcb, op_code, len, p_data);
- break;
- default:
- GATT_TRACE_ERROR("Unknown opcode = %d", op_code);
- break;
- }
- }
- if (op_code != GATT_HANDLE_VALUE_IND && op_code != GATT_HANDLE_VALUE_NOTIF) {
- gatt_cl_send_next_cmd_inq(p_tcb);
- }
- return;
- }
- #endif /* BLE_INCLUDED == TRUE && GATTC_INCLUDED == TRUE */
|