1 /******************************************************************************
2 *
3 * Copyright (C) 1999-2012 Broadcom Corporation
4 *
5 * Licensed under the Apache License, Version 2.0 (the "License");
6 * you may not use this file except in compliance with the License.
7 * You may obtain a copy of the License at:
8 *
9 * http://www.apache.org/licenses/LICENSE-2.0
10 *
11 * Unless required by applicable law or agreed to in writing, software
12 * distributed under the License is distributed on an "AS IS" BASIS,
13 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14 * See the License for the specific language governing permissions and
15 * limitations under the License.
16 *
17 ******************************************************************************/
18
19 /******************************************************************************
20 *
21 * this file contains GATT interface functions
22 *
23 ******************************************************************************/
24 #include "common/bt_target.h"
25
26
27 #if defined(BTA_GATT_INCLUDED) && (BTA_GATT_INCLUDED == 1)
28
29 #include "osi/allocator.h"
30 #include <string.h>
31 #include "stack/gatt_api.h"
32 #include "gatt_int.h"
33 #include "stack/l2c_api.h"
34 #include "btm_int.h"
35 #include "stack/sdpdefs.h"
36 #include "stack/sdp_api.h"
37
38 /*******************************************************************************
39 **
40 ** Function GATT_SetTraceLevel
41 **
42 ** Description This function sets the trace level. If called with
43 ** a value of 0xFF, it simply returns the current trace level.
44 **
45 ** Input Parameters:
46 ** level: The level to set the GATT tracing to:
47 ** 0xff-returns the current setting.
48 ** 0-turns off tracing.
49 ** >= 1-Errors.
50 ** >= 2-Warnings.
51 ** >= 3-APIs.
52 ** >= 4-Events.
53 ** >= 5-Debug.
54 **
55 ** Returns The new or current trace level
56 **
57 *******************************************************************************/
GATT_SetTraceLevel(UINT8 new_level)58 UINT8 GATT_SetTraceLevel (UINT8 new_level)
59 {
60 if (new_level != 0xFF) {
61 gatt_cb.trace_level = new_level;
62 }
63
64 return (gatt_cb.trace_level);
65 }
66
67
68 #if (GATTS_INCLUDED == 1)
69 /*****************************************************************************
70 **
71 ** GATT SERVER API
72 **
73 ******************************************************************************/
74 /*******************************************************************************
75 **
76 ** Function GATTS_AddHandleRange
77 **
78 ** Description This function add the allocated handles range for the specifed
79 ** application UUID, service UUID and service instance
80 **
81 ** Parameter p_hndl_range: pointer to allocated handles information
82 **
83 ** Returns 1 if handle range is added successfully; otherwise 0.
84 **
85 *******************************************************************************/
86
GATTS_AddHandleRange(tGATTS_HNDL_RANGE * p_hndl_range)87 BOOLEAN GATTS_AddHandleRange(tGATTS_HNDL_RANGE *p_hndl_range)
88 {
89 tGATT_HDL_LIST_ELEM *p_buf;
90 BOOLEAN status = 0;
91
92 if ((p_buf = gatt_alloc_hdl_buffer()) != NULL) {
93 p_buf->asgn_range = *p_hndl_range;
94 status = gatt_add_an_item_to_list(&gatt_cb.hdl_list_info, p_buf);
95 }
96 return status;
97 }
98
99
100 /*******************************************************************************
101 **
102 ** Function GATTS_NVRegister
103 **
104 ** Description Application manager calls this function to register for
105 ** NV save callback function. There can be one and only one
106 ** NV save callback function.
107 **
108 ** Parameter p_cb_info : callback informaiton
109 **
110 ** Returns 1 if registered OK, else 0
111 **
112 *******************************************************************************/
GATTS_NVRegister(const tGATT_APPL_INFO * p_cb_info)113 BOOLEAN GATTS_NVRegister (const tGATT_APPL_INFO *p_cb_info)
114 {
115 BOOLEAN status = 0;
116 if (p_cb_info) {
117 gatt_cb.cb_info = *p_cb_info;
118 status = 1;
119 gatt_init_srv_chg();
120 }
121
122 return status;
123 }
124
125 /*******************************************************************************
126 **
127 ** Function GATTS_CreateService
128 **
129 ** Description This function is called to reserve a block of handles for a service.
130 **
131 ** *** It should be called only once per service instance ***
132 **
133 ** Parameter gatt_if : application if
134 ** p_svc_uuid : service UUID
135 ** svc_inst : instance of the service inside the application
136 ** num_handles : number of handles needed by the service.
137 ** is_pri : is a primary service or not.
138 **
139 ** Returns service handle if sucessful, otherwise 0.
140 **
141 *******************************************************************************/
GATTS_CreateService(tGATT_IF gatt_if,tBT_UUID * p_svc_uuid,UINT16 svc_inst,UINT16 num_handles,BOOLEAN is_pri)142 UINT16 GATTS_CreateService (tGATT_IF gatt_if, tBT_UUID *p_svc_uuid,
143 UINT16 svc_inst, UINT16 num_handles, BOOLEAN is_pri)
144 {
145
146 tGATT_HDL_LIST_INFO *p_list_info = &gatt_cb.hdl_list_info;
147 tGATT_HDL_LIST_ELEM *p_list = NULL;
148 UINT16 s_hdl = 0;
149 BOOLEAN save_hdl = 0;
150 tGATTS_PENDING_NEW_SRV_START *p_buf = NULL;
151 tGATT_REG *p_reg = gatt_get_regcb(gatt_if);
152 tBT_UUID *p_app_uuid128;
153
154
155 GATT_TRACE_API ("GATTS_CreateService\n" );
156
157 if (p_reg == NULL) {
158 GATT_TRACE_ERROR ("Inavlid gatt_if=%d\n", gatt_if);
159 return (0);
160 }
161
162 p_app_uuid128 = &p_reg->app_uuid128;
163
164 if ((p_list = gatt_find_hdl_buffer_by_app_id(p_app_uuid128, p_svc_uuid, svc_inst)) != NULL) {
165 s_hdl = p_list->asgn_range.s_handle;
166 GATT_TRACE_DEBUG ("Service already been created!!\n");
167 } else {
168 if ( (p_svc_uuid->len == LEN_UUID_16) && (p_svc_uuid->uu.uuid16 == UUID_SERVCLASS_GATT_SERVER)) {
169 s_hdl = gatt_cb.hdl_cfg.gatt_start_hdl;
170 } else if ((p_svc_uuid->len == LEN_UUID_16) && (p_svc_uuid->uu.uuid16 == UUID_SERVCLASS_GAP_SERVER)) {
171 s_hdl = gatt_cb.hdl_cfg.gap_start_hdl;
172 } else {
173 p_list = p_list_info->p_first;
174
175 if (p_list) {
176 s_hdl = p_list->asgn_range.e_handle + 1;
177 }
178
179 if (s_hdl < gatt_cb.hdl_cfg.app_start_hdl) {
180
181 s_hdl = gatt_cb.hdl_cfg.app_start_hdl;
182 }
183 save_hdl = 1;
184 }
185
186 /* check for space */
187 if (num_handles > (0xFFFF - s_hdl + 1)) {
188 GATT_TRACE_ERROR ("GATTS_ReserveHandles: no handles, s_hdl: %u needed: %u\n", s_hdl, num_handles);
189 return (0);
190 }
191
192 if ( (p_list = gatt_alloc_hdl_buffer()) == NULL) {
193 /* No free entry */
194 GATT_TRACE_ERROR ("GATTS_ReserveHandles: no free handle blocks\n");
195 return (0);
196 }
197
198 p_list->asgn_range.app_uuid128 = *p_app_uuid128;
199 p_list->asgn_range.svc_uuid = *p_svc_uuid;
200 p_list->asgn_range.svc_inst = svc_inst;
201 p_list->asgn_range.s_handle = s_hdl;
202 p_list->asgn_range.e_handle = s_hdl + num_handles - 1;
203 p_list->asgn_range.is_primary = is_pri;
204
205 gatt_add_an_item_to_list(p_list_info, p_list);
206
207 if (save_hdl) {
208 if (gatt_cb.cb_info.p_nv_save_callback) {
209 (*gatt_cb.cb_info.p_nv_save_callback)(1, &p_list->asgn_range);
210 }
211 /* add a pending new service change item to the list */
212 if ( (p_buf = gatt_add_pending_new_srv_start(&p_list->asgn_range)) == NULL) {
213 /* No free entry */
214 GATT_TRACE_ERROR ("gatt_add_pending_new_srv_start: no free blocks\n");
215
216 if (p_list) {
217 gatt_remove_an_item_from_list(p_list_info, p_list);
218 gatt_free_attr_value_buffer(p_list);
219 gatt_free_hdl_buffer(p_list);
220 }
221 return (0);
222 }
223
224 GATT_TRACE_DEBUG ("Add a new srv chg item\n");
225 }
226 }
227
228 if (!gatts_init_service_db(&p_list->svc_db, p_svc_uuid, is_pri, s_hdl , num_handles)) {
229 GATT_TRACE_ERROR ("GATTS_ReserveHandles: service DB initialization failed\n");
230 if (p_list) {
231 gatt_remove_an_item_from_list(p_list_info, p_list);
232 gatt_free_attr_value_buffer(p_list);
233 gatt_free_hdl_buffer(p_list);
234 }
235
236 if (p_buf) {
237 osi_free(fixed_queue_try_remove_from_queue(gatt_cb.pending_new_srv_start_q, p_buf));
238 }
239 return (0);
240 }
241
242 return (s_hdl);
243 }
244
245 /*******************************************************************************
246 **
247 ** Function GATTS_AddIncludeService
248 **
249 ** Description This function is called to add an included service.
250 **
251 ** Parameter service_handle : To which service this included service is added to.
252 ** include_svc_handle : included service handle.
253 **
254 ** Returns included service attribute handle. If 0, add included service
255 ** fail.
256 **
257 *******************************************************************************/
GATTS_AddIncludeService(UINT16 service_handle,UINT16 include_svc_handle)258 UINT16 GATTS_AddIncludeService (UINT16 service_handle, UINT16 include_svc_handle)
259
260 {
261 tGATT_HDL_LIST_ELEM *p_decl, *p_incl_decl;
262
263 if ((p_decl = gatt_find_hdl_buffer_by_handle(service_handle)) == NULL) {
264 GATT_TRACE_DEBUG("Service not created");
265 return 0;
266 }
267 if ((p_incl_decl = gatt_find_hdl_buffer_by_handle(include_svc_handle)) == NULL) {
268 GATT_TRACE_DEBUG("Included Service not created");
269 return 0;
270 }
271
272 return gatts_add_included_service(&p_decl->svc_db,
273 p_incl_decl->asgn_range.s_handle,
274 p_incl_decl->asgn_range.e_handle,
275 p_incl_decl->asgn_range.svc_uuid);
276 }
277 /*******************************************************************************
278 **
279 ** Function GATTS_AddCharacteristic
280 **
281 ** Description This function is called to add a characteristic into a service.
282 ** It will add a characteristic declaration and characteristic
283 ** value declaration into the service database identified by the
284 ** service handle.
285 **
286 ** Parameter service_handle : To which service this included service is added to.
287 ** char_uuid : Characteristic UUID.
288 ** perm : Characteristic value declaration attribute permission.
289 ** property : Characteristic Properties
290 **
291 ** Returns Characteristic value declaration attribute handle. 0 if failed.
292 **
293 *******************************************************************************/
GATTS_AddCharacteristic(UINT16 service_handle,tBT_UUID * p_char_uuid,tGATT_PERM perm,tGATT_CHAR_PROP property,tGATT_ATTR_VAL * attr_val,tGATTS_ATTR_CONTROL * control)294 UINT16 GATTS_AddCharacteristic (UINT16 service_handle, tBT_UUID *p_char_uuid,
295 tGATT_PERM perm, tGATT_CHAR_PROP property,
296 tGATT_ATTR_VAL *attr_val, tGATTS_ATTR_CONTROL *control)
297 {
298 tGATT_HDL_LIST_ELEM *p_decl;
299
300 if ((p_decl = gatt_find_hdl_buffer_by_handle(service_handle)) == NULL) {
301 GATT_TRACE_DEBUG("Service not created\n");
302 return 0;
303 }
304 /* data validity checking */
305 if ( ((property & GATT_CHAR_PROP_BIT_AUTH) && !(perm & GATT_WRITE_SIGNED_PERM)) ||
306 ((perm & GATT_WRITE_SIGNED_PERM) && !(property & GATT_CHAR_PROP_BIT_AUTH)) ) {
307 GATT_TRACE_DEBUG("Invalid configuration property=0x%x perm=0x%x\n ", property, perm);
308 return 0;
309 }
310
311 return gatts_add_characteristic(&p_decl->svc_db,
312 perm,
313 property,
314 p_char_uuid,
315 attr_val, control);
316 }
317 /*******************************************************************************
318 **
319 ** Function GATTS_AddCharDescriptor
320 **
321 ** Description This function is called to add a characteristic descriptor
322 ** into a service database. Add descriptor should follow add char
323 ** to which it belongs, and next add char should be done only
324 ** after all add descriptors for the previous char.
325 **
326 ** Parameter service_handle : To which service this characteristic descriptor
327 ** is added to.
328 ** perm : Characteristic value declaration attribute
329 ** permission.
330 ** p_descr_uuid : Characteristic descriptor UUID
331 **
332 ** Returns Characteristic descriptor attribute handle. 0 if add
333 ** characteristic descriptor failed.
334 **
335 *******************************************************************************/
GATTS_AddCharDescriptor(UINT16 service_handle,tGATT_PERM perm,tBT_UUID * p_descr_uuid,tGATT_ATTR_VAL * attr_val,tGATTS_ATTR_CONTROL * control)336 UINT16 GATTS_AddCharDescriptor (UINT16 service_handle,
337 tGATT_PERM perm,
338 tBT_UUID *p_descr_uuid, tGATT_ATTR_VAL *attr_val, tGATTS_ATTR_CONTROL *control)
339 {
340 tGATT_HDL_LIST_ELEM *p_decl;
341
342 if ((p_decl = gatt_find_hdl_buffer_by_handle(service_handle)) == NULL) {
343 GATT_TRACE_DEBUG("Service not created");
344 return 0;
345 }
346 if (p_descr_uuid == NULL ||
347 (p_descr_uuid->len != LEN_UUID_128 && p_descr_uuid->len != LEN_UUID_16
348 && p_descr_uuid->len != LEN_UUID_32)) {
349 GATT_TRACE_DEBUG("Illegal parameter");
350 return 0;
351 }
352
353 return gatts_add_char_descr(&p_decl->svc_db,
354 perm,
355 p_descr_uuid,
356 attr_val, control);
357
358 }
359 /*******************************************************************************
360 **
361 ** Function GATTS_DeleteService
362 **
363 ** Description This function is called to delete a service.
364 **
365 ** Parameter gatt_if : application interface
366 ** p_svc_uuid : service UUID
367 ** svc_inst : instance of the service inside the application
368 **
369 ** Returns 1 if operation succeed, 0 if handle block was not found.
370 **
371 *******************************************************************************/
GATTS_DeleteService(tGATT_IF gatt_if,tBT_UUID * p_svc_uuid,UINT16 svc_inst)372 BOOLEAN GATTS_DeleteService (tGATT_IF gatt_if, tBT_UUID *p_svc_uuid, UINT16 svc_inst)
373 {
374
375 tGATT_HDL_LIST_INFO *p_list_info = &gatt_cb.hdl_list_info;
376 tGATT_HDL_LIST_ELEM *p_list = NULL;
377 UINT8 i_sreg;
378 tGATTS_PENDING_NEW_SRV_START *p_buf;
379 tGATT_REG *p_reg = gatt_get_regcb(gatt_if);
380 tBT_UUID *p_app_uuid128;
381
382 GATT_TRACE_DEBUG ("GATTS_DeleteService");
383
384 if (p_reg == NULL) {
385 GATT_TRACE_ERROR ("Application not foud");
386 return (0);
387 }
388 p_app_uuid128 = &p_reg->app_uuid128;
389
390 if ((p_list = gatt_find_hdl_buffer_by_app_id(p_app_uuid128, p_svc_uuid, svc_inst)) == NULL) {
391 GATT_TRACE_ERROR ("No Service found");
392 return (0);
393 }
394
395 if ( (p_buf = gatt_sr_is_new_srv_chg(&p_list->asgn_range.app_uuid128,
396 &p_list->asgn_range.svc_uuid,
397 p_list->asgn_range.svc_inst)) != NULL) {
398 GATT_TRACE_DEBUG ("Delete a new service changed item - the service has not yet started");
399 osi_free(fixed_queue_try_remove_from_queue(gatt_cb.pending_new_srv_start_q, p_buf));
400 } else {
401 if (GATTS_SEND_SERVICE_CHANGE_MODE == GATTS_SEND_SERVICE_CHANGE_AUTO) {
402 gatt_proc_srv_chg();
403 }
404 }
405
406 if ((i_sreg = gatt_sr_find_i_rcb_by_app_id (p_app_uuid128,
407 p_svc_uuid,
408 svc_inst)) != GATT_MAX_SR_PROFILES) {
409 GATTS_StopService(gatt_cb.sr_reg[i_sreg].s_hdl);
410 }
411
412 GATT_TRACE_DEBUG ("released handles s_hdl=%u e_hdl=%u",
413 p_list->asgn_range.s_handle , p_list->asgn_range.e_handle );
414
415 if ( (p_list->asgn_range.s_handle >= gatt_cb.hdl_cfg.app_start_hdl)
416 && gatt_cb.cb_info.p_nv_save_callback) {
417 (*gatt_cb.cb_info.p_nv_save_callback)(0, &p_list->asgn_range);
418 }
419
420 gatt_remove_an_item_from_list(p_list_info, p_list);
421 gatt_free_attr_value_buffer(p_list);
422 gatt_free_hdl_buffer(p_list);
423
424 return (1);
425 }
426
427 /*******************************************************************************
428 **
429 ** Function GATTS_StartService
430 **
431 ** Description This function is called to start a service with GATT
432 **
433 ** Parameter gatt_if : service handle.
434 ** p_cback : application service callback functions.
435 ** sup_transport : supported transport(s) for this primary service
436 **
437 ** return GATT_SUCCESS if successfully started; otherwise error code.
438 **
439 *******************************************************************************/
GATTS_StartService(tGATT_IF gatt_if,UINT16 service_handle,tGATT_TRANSPORT sup_transport)440 tGATT_STATUS GATTS_StartService (tGATT_IF gatt_if, UINT16 service_handle,
441 tGATT_TRANSPORT sup_transport)
442 {
443 tGATT_SR_REG *p_sreg;
444 tGATT_HDL_LIST_ELEM *p_list = NULL;
445 UINT8 i_sreg;
446 #if (SDP_INCLUDED == 1 && CLASSIC_BT_GATT_INCLUDED == 1)
447 tBT_UUID *p_uuid;
448 #endif ///SDP_INCLUDED == 1 && CLASSIC_BT_GATT_INCLUDED == 1
449 tGATT_REG *p_reg = gatt_get_regcb(gatt_if);
450
451 tGATTS_PENDING_NEW_SRV_START *p_buf;
452
453 GATT_TRACE_API ("GATTS_StartService");
454
455 if (p_reg == NULL) {
456 /* Not found */
457 GATT_TRACE_ERROR ("Application not found ");
458 return GATT_NOT_FOUND;
459 }
460
461 if ((p_list = gatt_find_hdl_buffer_by_handle(service_handle)) == NULL) {
462 /* Not found */
463 GATT_TRACE_ERROR ("no service found");
464 return GATT_NOT_FOUND;
465 }
466
467 if (gatt_sr_find_i_rcb_by_app_id (&p_list->asgn_range.app_uuid128,
468 &p_list->asgn_range.svc_uuid,
469 p_list->asgn_range.svc_inst) != GATT_MAX_SR_PROFILES) {
470 GATT_TRACE_ERROR ("Duplicate Service start - Service already started");
471 return GATT_SERVICE_STARTED;
472 }
473
474 /*this is a new application servoce start */
475 if ((i_sreg = gatt_sr_alloc_rcb(p_list)) == GATT_MAX_SR_PROFILES) {
476 GATT_TRACE_ERROR ("GATTS_StartService: no free server registration block");
477 return GATT_NO_RESOURCES;
478 }
479
480 p_sreg = &gatt_cb.sr_reg[i_sreg];
481 p_sreg->gatt_if = gatt_if;
482
483 switch (sup_transport) {
484 case GATT_TRANSPORT_BR_EDR:
485 case GATT_TRANSPORT_LE_BR_EDR:
486 if (p_sreg->type == GATT_UUID_PRI_SERVICE) {
487 #if (SDP_INCLUDED == 1 && CLASSIC_BT_GATT_INCLUDED == 1)
488 p_uuid = gatts_get_service_uuid (p_sreg->p_db);
489 p_sreg->sdp_handle = gatt_add_sdp_record(p_uuid, p_sreg->s_hdl, p_sreg->e_hdl);
490 #endif ///SDP_INCLUDED == 1 && CLASSIC_BT_GATT_INCLUDED == 1
491 }
492 break;
493 default:
494 break;
495 }
496
497 gatts_update_srv_list_elem(i_sreg, p_sreg->s_hdl,
498 p_list->asgn_range.is_primary);
499
500 gatt_add_a_srv_to_list(&gatt_cb.srv_list_info, &gatt_cb.srv_list[i_sreg]);
501
502 GATT_TRACE_DEBUG ("allocated i_sreg=%d\n", i_sreg);
503
504 GATT_TRACE_DEBUG ("s_hdl=%d e_hdl=%d type=0x%x svc_inst=%d sdp_hdl=0x%x\n",
505 p_sreg->s_hdl, p_sreg->e_hdl,
506 p_sreg->type, p_sreg->service_instance,
507 p_sreg->sdp_handle);
508
509
510 if ( (p_buf = gatt_sr_is_new_srv_chg(&p_list->asgn_range.app_uuid128,
511 &p_list->asgn_range.svc_uuid,
512 p_list->asgn_range.svc_inst)) != NULL) {
513 if (GATTS_SEND_SERVICE_CHANGE_MODE == GATTS_SEND_SERVICE_CHANGE_AUTO) {
514 gatt_proc_srv_chg();
515 }
516 /* remove the new service element after the srv changed processing is completed*/
517
518 osi_free(fixed_queue_try_remove_from_queue(gatt_cb.pending_new_srv_start_q, p_buf));
519 }
520 return GATT_SUCCESS;
521 }
522
523 /*******************************************************************************
524 **
525 ** Function GATTS_StopService
526 **
527 ** Description This function is called to stop a service
528 **
529 ** Parameter service_handle : this is the start handle of a service
530 **
531 ** Returns None.
532 **
533 *******************************************************************************/
GATTS_StopService(UINT16 service_handle)534 void GATTS_StopService (UINT16 service_handle)
535 {
536 UINT8 ii = gatt_sr_find_i_rcb_by_handle(service_handle);
537
538 GATT_TRACE_API("GATTS_StopService %u", service_handle);
539
540 /* Index 0 is reserved for GATT, and is never stopped */
541 if ( (ii > 0) && (ii < GATT_MAX_SR_PROFILES) && (gatt_cb.sr_reg[ii].in_use) ) {
542 #if(SDP_INCLUDED == 1 && CLASSIC_BT_GATT_INCLUDED == 1)
543 if (gatt_cb.sr_reg[ii].sdp_handle) {
544 SDP_DeleteRecord(gatt_cb.sr_reg[ii].sdp_handle);
545 }
546 #endif ///SDP_INCLUDED == 1 && CLASSIC_BT_GATT_INCLUDED == 1
547 gatt_remove_a_srv_from_list(&gatt_cb.srv_list_info, &gatt_cb.srv_list[ii]);
548 gatt_cb.srv_list[ii].in_use = 0;
549 memset (&gatt_cb.sr_reg[ii], 0, sizeof(tGATT_SR_REG));
550 } else {
551 GATT_TRACE_ERROR("GATTS_StopService service_handle: %u is not in use", service_handle);
552 }
553 }
554 /*******************************************************************************
555 **
556 ** Function GATTs_HandleValueIndication
557 **
558 ** Description This function sends a handle value indication to a client.
559 **
560 ** Parameter conn_id: connection identifier.
561 ** attr_handle: Attribute handle of this handle value indication.
562 ** val_len: Length of the indicated attribute value.
563 ** p_val: Pointer to the indicated attribute value data.
564 **
565 ** Returns GATT_SUCCESS if successfully sent or queued; otherwise error code.
566 **
567 *******************************************************************************/
GATTS_HandleValueIndication(UINT16 conn_id,UINT16 attr_handle,UINT16 val_len,UINT8 * p_val)568 tGATT_STATUS GATTS_HandleValueIndication (UINT16 conn_id, UINT16 attr_handle, UINT16 val_len, UINT8 *p_val)
569 {
570 tGATT_STATUS cmd_status = GATT_NO_RESOURCES;
571
572 tGATT_VALUE indication;
573 BT_HDR *p_msg;
574 tGATT_IF gatt_if = GATT_GET_GATT_IF(conn_id);
575 UINT8 tcb_idx = GATT_GET_TCB_IDX(conn_id);
576 tGATT_REG *p_reg = gatt_get_regcb(gatt_if);
577 tGATT_TCB *p_tcb = gatt_get_tcb_by_idx(tcb_idx);
578
579
580 GATT_TRACE_API ("GATTS_HandleValueIndication");
581 if ( (p_reg == NULL) || (p_tcb == NULL)) {
582 GATT_TRACE_ERROR ("GATTS_HandleValueIndication Unknown conn_id: %u ", conn_id);
583 return (tGATT_STATUS) GATT_INVALID_CONN_ID;
584 }
585
586 if (! GATT_HANDLE_IS_VALID (attr_handle)) {
587 return GATT_ILLEGAL_PARAMETER;
588 }
589
590 indication.conn_id = conn_id;
591 indication.handle = attr_handle;
592 indication.len = val_len;
593 memcpy (indication.value, p_val, val_len);
594 indication.auth_req = GATT_AUTH_REQ_NONE;
595
596 if (GATT_HANDLE_IS_VALID(p_tcb->indicate_handle)) {
597 /* TODO: need to further check whether deleting pending queue here cause reducing transport performance */
598 /*
599 GATT_TRACE_DEBUG ("Add a pending indication");
600 if ((p_buf = gatt_add_pending_ind(p_tcb, &indication)) != NULL) {
601 cmd_status = GATT_SUCCESS;
602 } else {
603 cmd_status = GATT_NO_RESOURCES;
604 }
605 */
606 return GATT_BUSY;
607 } else {
608
609 if ( (p_msg = attp_build_sr_msg (p_tcb, GATT_HANDLE_VALUE_IND, (tGATT_SR_MSG *)&indication)) != NULL) {
610 cmd_status = attp_send_sr_msg (p_tcb, p_msg);
611
612 if (cmd_status == GATT_SUCCESS || cmd_status == GATT_CONGESTED) {
613 p_tcb->indicate_handle = indication.handle;
614 gatt_start_conf_timer(p_tcb);
615 }
616 }
617 }
618 return cmd_status;
619 }
620
621 /*******************************************************************************
622 **
623 ** Function GATTS_HandleValueNotification
624 **
625 ** Description This function sends a handle value notification to a client.
626 **
627 ** Parameter conn_id: connection identifier.
628 ** attr_handle: Attribute handle of this handle value indication.
629 ** val_len: Length of the indicated attribute value.
630 ** p_val: Pointer to the indicated attribute value data.
631 **
632 ** Returns GATT_SUCCESS if successfully sent; otherwise error code.
633 **
634 *******************************************************************************/
GATTS_HandleValueNotification(UINT16 conn_id,UINT16 attr_handle,UINT16 val_len,UINT8 * p_val)635 tGATT_STATUS GATTS_HandleValueNotification (UINT16 conn_id, UINT16 attr_handle,
636 UINT16 val_len, UINT8 *p_val)
637 {
638 tGATT_STATUS cmd_sent = GATT_ILLEGAL_PARAMETER;
639 BT_HDR *p_buf;
640 tGATT_VALUE notif;
641 tGATT_IF gatt_if = GATT_GET_GATT_IF(conn_id);
642 UINT8 tcb_idx = GATT_GET_TCB_IDX(conn_id);
643 tGATT_REG *p_reg = gatt_get_regcb(gatt_if);
644 tGATT_TCB *p_tcb = gatt_get_tcb_by_idx(tcb_idx);
645
646 GATT_TRACE_API ("GATTS_HandleValueNotification");
647
648 if ( (p_reg == NULL) || (p_tcb == NULL)) {
649 GATT_TRACE_ERROR ("GATTS_HandleValueNotification Unknown conn_id: %u \n", conn_id);
650 return (tGATT_STATUS) GATT_INVALID_CONN_ID;
651 }
652
653 if (GATT_HANDLE_IS_VALID (attr_handle)) {
654 notif.handle = attr_handle;
655 notif.len = val_len;
656 memcpy (notif.value, p_val, val_len);
657 notif.auth_req = GATT_AUTH_REQ_NONE;
658
659 if ((p_buf = attp_build_sr_msg (p_tcb, GATT_HANDLE_VALUE_NOTIF, (tGATT_SR_MSG *)¬if))
660 != NULL) {
661 cmd_sent = attp_send_sr_msg (p_tcb, p_buf);
662 } else {
663 cmd_sent = GATT_NO_RESOURCES;
664 }
665 }
666 return cmd_sent;
667 }
668
669 /*******************************************************************************
670 **
671 ** Function GATTS_SendRsp
672 **
673 ** Description This function sends the server response to client.
674 **
675 ** Parameter conn_id: connection identifier.
676 ** trans_id: transaction id
677 ** status: response status
678 ** p_msg: pointer to message parameters structure.
679 **
680 ** Returns GATT_SUCCESS if successfully sent; otherwise error code.
681 **
682 *******************************************************************************/
GATTS_SendRsp(UINT16 conn_id,UINT32 trans_id,tGATT_STATUS status,tGATTS_RSP * p_msg)683 tGATT_STATUS GATTS_SendRsp (UINT16 conn_id, UINT32 trans_id,
684 tGATT_STATUS status, tGATTS_RSP *p_msg)
685 {
686 tGATT_STATUS cmd_sent = GATT_ILLEGAL_PARAMETER;
687 tGATT_IF gatt_if = GATT_GET_GATT_IF(conn_id);
688 UINT8 tcb_idx = GATT_GET_TCB_IDX(conn_id);
689 tGATT_REG *p_reg = gatt_get_regcb(gatt_if);
690 tGATT_TCB *p_tcb = gatt_get_tcb_by_idx(tcb_idx);
691
692 GATT_TRACE_API ("GATTS_SendRsp: conn_id: %u trans_id: %u Status: 0x%04x\n",
693 conn_id, trans_id, status);
694
695 if ( (p_reg == NULL) || (p_tcb == NULL)) {
696 GATT_TRACE_ERROR ("GATTS_SendRsp Unknown conn_id: %u\n", conn_id);
697 return (tGATT_STATUS) GATT_INVALID_CONN_ID;
698 }
699
700 if (p_tcb->sr_cmd.trans_id != trans_id) {
701 GATT_TRACE_ERROR ("GATTS_SendRsp conn_id: %u waiting for op_code = %02x\n",
702 conn_id, p_tcb->sr_cmd.op_code);
703
704 return (GATT_WRONG_STATE);
705 }
706 /* Process App response */
707 cmd_sent = gatt_sr_process_app_rsp (p_tcb, gatt_if, trans_id, p_tcb->sr_cmd.op_code, status, p_msg);
708
709 return cmd_sent;
710 }
711
712
713 /*******************************************************************************
714 **
715 ** Function GATTS_SetAttributeValue
716 **
717 ** Description This function sends to set the attribute value .
718 **
719 ** Parameter attr_handle:the attribute handle
720 ** length: the attribute length
721 ** value: the value to be set to the attribute in the database
722 **
723 ** Returns GATT_SUCCESS if successfully sent; otherwise error code.
724 **
725 *******************************************************************************/
GATTS_SetAttributeValue(UINT16 attr_handle,UINT16 length,UINT8 * value)726 tGATT_STATUS GATTS_SetAttributeValue(UINT16 attr_handle, UINT16 length, UINT8 *value)
727 {
728 tGATT_STATUS status;
729 tGATT_HDL_LIST_ELEM *p_decl = NULL;
730
731 GATT_TRACE_DEBUG("GATTS_SetAttributeValue: attr_handle: %u length: %u \n",
732 attr_handle, length);
733 if (length <= 0){
734 return GATT_INVALID_ATTR_LEN;
735 }
736 if ((p_decl = gatt_find_hdl_buffer_by_attr_handle(attr_handle)) == NULL) {
737 GATT_TRACE_DEBUG("Service not created\n");
738 return GATT_INVALID_HANDLE;
739 }
740
741 status = gatts_set_attribute_value(&p_decl->svc_db, attr_handle, length, value);
742 return status;
743
744 }
745
746
747 /*******************************************************************************
748 **
749 ** Function GATTS_GetAttributeValue
750 **
751 ** Description This function sends to set the attribute value .
752 **
753 ** Parameter attr_handle: the attribute handle
754 ** length:the attribute value length in the database
755 ** value: the attribute value out put
756 **
757 ** Returns GATT_SUCCESS if successfully sent; otherwise error code.
758 **
759 *******************************************************************************/
GATTS_GetAttributeValue(UINT16 attr_handle,UINT16 * length,UINT8 ** value)760 tGATT_STATUS GATTS_GetAttributeValue(UINT16 attr_handle, UINT16 *length, UINT8 **value)
761 {
762 tGATT_STATUS status;
763 tGATT_HDL_LIST_ELEM *p_decl;
764
765 GATT_TRACE_DEBUG("GATTS_GetAttributeValue: attr_handle: %u\n",
766 attr_handle);
767
768 if ((p_decl = gatt_find_hdl_buffer_by_attr_handle(attr_handle)) == NULL) {
769 GATT_TRACE_ERROR("Service not created\n");
770 *length = 0;
771 return GATT_INVALID_HANDLE;
772 }
773
774 status = gatts_get_attribute_value(&p_decl->svc_db, attr_handle, length, value);
775 return status;
776 }
777 #endif ///GATTS_INCLUDED == 1
778
779
780 #if (GATTC_INCLUDED == 1)
781 /*******************************************************************************/
782 /* GATT Profile Srvr Functions */
783 /*******************************************************************************/
784
785 /*******************************************************************************/
786 /* */
787 /* GATT CLIENT APIs */
788 /* */
789 /*******************************************************************************/
790
791
792 /*******************************************************************************
793 **
794 ** Function GATTC_ConfigureMTU
795 **
796 ** Description This function is called to configure the ATT MTU size.
797 **
798 ** Parameters conn_id: connection identifier.
799 ** mtu - attribute MTU size..
800 **
801 ** Returns GATT_SUCCESS if command started successfully.
802 **
803 *******************************************************************************/
GATTC_ConfigureMTU(UINT16 conn_id)804 tGATT_STATUS GATTC_ConfigureMTU (UINT16 conn_id)
805 {
806 UINT8 ret = GATT_NO_RESOURCES;
807 tGATT_IF gatt_if = GATT_GET_GATT_IF(conn_id);
808 UINT8 tcb_idx = GATT_GET_TCB_IDX(conn_id);
809 tGATT_TCB *p_tcb = gatt_get_tcb_by_idx(tcb_idx);
810 tGATT_REG *p_reg = gatt_get_regcb(gatt_if);
811
812 tGATT_CLCB *p_clcb;
813 uint16_t mtu = gatt_get_local_mtu();
814
815 GATT_TRACE_API ("GATTC_ConfigureMTU conn_id=%d mtu=%d", conn_id, mtu );
816
817 /* Validate that the link is BLE, not BR/EDR */
818 if (p_tcb->transport != BT_TRANSPORT_LE) {
819 return GATT_ERROR;
820 }
821
822 if ( (p_tcb == NULL) || (p_reg == NULL) || (mtu < GATT_DEF_BLE_MTU_SIZE) || (mtu > GATT_MAX_MTU_SIZE)) {
823 return GATT_ILLEGAL_PARAMETER;
824 }
825
826 if (gatt_is_clcb_allocated(conn_id)) {
827 GATT_TRACE_ERROR("GATTC_ConfigureMTU GATT_BUSY conn_id = %d", conn_id);
828 return GATT_BUSY;
829 }
830
831 if ((p_clcb = gatt_clcb_alloc(conn_id)) != NULL) {
832 p_clcb->p_tcb->payload_size = mtu;
833 p_clcb->operation = GATTC_OPTYPE_CONFIG;
834
835 ret = attp_send_cl_msg (p_clcb->p_tcb, p_clcb->clcb_idx, GATT_REQ_MTU, (tGATT_CL_MSG *)&mtu);
836 }
837
838 return ret;
839 }
840
841 /*******************************************************************************
842 **
843 ** Function GATTC_Discover
844 **
845 ** Description This function is called to do a discovery procedure on ATT server.
846 **
847 ** Parameters conn_id: connection identifier.
848 ** disc_type:discovery type.
849 ** p_param: parameters of discovery requirement.
850 **
851 ** Returns GATT_SUCCESS if command received/sent successfully.
852 **
853 *******************************************************************************/
GATTC_Discover(UINT16 conn_id,tGATT_DISC_TYPE disc_type,tGATT_DISC_PARAM * p_param)854 tGATT_STATUS GATTC_Discover (UINT16 conn_id, tGATT_DISC_TYPE disc_type,
855 tGATT_DISC_PARAM *p_param)
856 {
857 tGATT_STATUS status = GATT_SUCCESS;
858 tGATT_CLCB *p_clcb;
859 tGATT_IF gatt_if = GATT_GET_GATT_IF(conn_id);
860 UINT8 tcb_idx = GATT_GET_TCB_IDX(conn_id);
861 tGATT_TCB *p_tcb = gatt_get_tcb_by_idx(tcb_idx);
862 tGATT_REG *p_reg = gatt_get_regcb(gatt_if);
863
864
865 GATT_TRACE_API ("GATTC_Discover conn_id=%d disc_type=%d", conn_id, disc_type);
866
867 if ( (p_tcb == NULL) || (p_reg == NULL) || (p_param == NULL) ||
868 (disc_type >= GATT_DISC_MAX)) {
869 GATT_TRACE_ERROR("GATTC_Discover Illegal param: disc_type %d conn_id = %d", disc_type, conn_id);
870 return GATT_ILLEGAL_PARAMETER;
871 }
872
873
874 if (gatt_is_clcb_allocated(conn_id)) {
875 GATT_TRACE_ERROR("GATTC_Discover GATT_BUSY conn_id = %d", conn_id);
876 return GATT_BUSY;
877 }
878
879
880 if ((p_clcb = gatt_clcb_alloc(conn_id)) != NULL ) {
881 if (!GATT_HANDLE_IS_VALID(p_param->s_handle) ||
882 !GATT_HANDLE_IS_VALID(p_param->e_handle) ||
883 /* search by type does not have a valid UUID param */
884 (disc_type == GATT_DISC_SRVC_BY_UUID &&
885 p_param->service.len == 0)) {
886 gatt_clcb_dealloc(p_clcb);
887 return GATT_ILLEGAL_PARAMETER;
888 }
889
890 p_clcb->operation = GATTC_OPTYPE_DISCOVERY;
891 p_clcb->op_subtype = disc_type;
892 p_clcb->s_handle = p_param->s_handle;
893 p_clcb->e_handle = p_param->e_handle;
894 p_clcb->uuid = p_param->service;
895
896 gatt_act_discovery(p_clcb);
897 } else {
898 status = GATT_NO_RESOURCES;
899 }
900 return status;
901 }
902
903 /*******************************************************************************
904 **
905 ** Function GATTC_Read
906 **
907 ** Description This function is called to read the value of an attribute from
908 ** the server.
909 **
910 ** Parameters conn_id: connection identifier.
911 ** type - attribute read type.
912 ** p_read - read operation parameters.
913 **
914 ** Returns GATT_SUCCESS if command started successfully.
915 **
916 *******************************************************************************/
GATTC_Read(UINT16 conn_id,tGATT_READ_TYPE type,tGATT_READ_PARAM * p_read)917 tGATT_STATUS GATTC_Read (UINT16 conn_id, tGATT_READ_TYPE type, tGATT_READ_PARAM *p_read)
918 {
919 tGATT_STATUS status = GATT_SUCCESS;
920 tGATT_CLCB *p_clcb;
921 tGATT_READ_MULTI *p_read_multi;
922 tGATT_IF gatt_if = GATT_GET_GATT_IF(conn_id);
923 UINT8 tcb_idx = GATT_GET_TCB_IDX(conn_id);
924 tGATT_TCB *p_tcb = gatt_get_tcb_by_idx(tcb_idx);
925 tGATT_REG *p_reg = gatt_get_regcb(gatt_if);
926
927
928 GATT_TRACE_API ("GATTC_Read conn_id=%d type=%d", conn_id, type);
929
930 if ( (p_tcb == NULL) || (p_reg == NULL) || (p_read == NULL) || ((type >= GATT_READ_MAX) || (type == 0))) {
931 GATT_TRACE_ERROR("GATT_Read Illegal param: conn_id %d, type 0%d,", conn_id, type);
932 return GATT_ILLEGAL_PARAMETER;
933 }
934
935 if (gatt_is_clcb_allocated(conn_id)) {
936 GATT_TRACE_ERROR("GATTC_Read GATT_BUSY conn_id = %d", conn_id);
937 return GATT_BUSY;
938 }
939
940 if ( (p_clcb = gatt_clcb_alloc(conn_id)) != NULL ) {
941 p_clcb->operation = GATTC_OPTYPE_READ;
942 p_clcb->op_subtype = type;
943 p_clcb->auth_req = p_read->by_handle.auth_req;
944 p_clcb->counter = 0;
945
946 switch (type) {
947 case GATT_READ_BY_TYPE:
948 case GATT_READ_CHAR_VALUE:
949 p_clcb->s_handle = p_read->service.s_handle;
950 p_clcb->e_handle = p_read->service.e_handle;
951 memcpy(&p_clcb->uuid, &p_read->service.uuid, sizeof(tBT_UUID));
952 break;
953 case GATT_READ_MULTIPLE:
954 p_clcb->s_handle = 0;
955 /* copy multiple handles in CB */
956 p_read_multi = (tGATT_READ_MULTI *)osi_malloc(sizeof(tGATT_READ_MULTI));
957 p_clcb->p_attr_buf = (UINT8 *)p_read_multi;
958 memcpy (p_read_multi, &p_read->read_multiple, sizeof(tGATT_READ_MULTI));
959 case GATT_READ_BY_HANDLE:
960 case GATT_READ_PARTIAL:
961 memset(&p_clcb->uuid, 0, sizeof(tBT_UUID));
962 p_clcb->s_handle = p_read->by_handle.handle;
963
964 if (type == GATT_READ_PARTIAL) {
965 p_clcb->counter = p_read->partial.offset;
966 }
967
968 break;
969 default:
970 break;
971 }
972 /* start security check */
973 if (gatt_security_check_start(p_clcb) == 0) {
974 status = GATT_NO_RESOURCES;
975 gatt_clcb_dealloc(p_clcb);
976 }
977 } else {
978 status = GATT_NO_RESOURCES;
979 }
980 return status;
981 }
982
983 /*******************************************************************************
984 **
985 ** Function GATTC_Write
986 **
987 ** Description This function is called to write the value of an attribute to
988 ** the server.
989 **
990 ** Parameters conn_id: connection identifier.
991 ** type - attribute write type.
992 ** p_write - write operation parameters.
993 **
994 ** Returns GATT_SUCCESS if command started successfully.
995 **
996 *******************************************************************************/
GATTC_Write(UINT16 conn_id,tGATT_WRITE_TYPE type,tGATT_VALUE * p_write)997 tGATT_STATUS GATTC_Write (UINT16 conn_id, tGATT_WRITE_TYPE type, tGATT_VALUE *p_write)
998 {
999 tGATT_STATUS status = GATT_SUCCESS;
1000 tGATT_CLCB *p_clcb;
1001 tGATT_VALUE *p;
1002 tGATT_IF gatt_if = GATT_GET_GATT_IF(conn_id);
1003 UINT8 tcb_idx = GATT_GET_TCB_IDX(conn_id);
1004 tGATT_TCB *p_tcb = gatt_get_tcb_by_idx(tcb_idx);
1005 tGATT_REG *p_reg = gatt_get_regcb(gatt_if);
1006
1007 if ( (p_tcb == NULL) || (p_reg == NULL) || (p_write == NULL) ||
1008 ((type != GATT_WRITE) && (type != GATT_WRITE_PREPARE) && (type != GATT_WRITE_NO_RSP)) ) {
1009 GATT_TRACE_ERROR("GATT_Write Illegal param: conn_id %d, type 0%d,", conn_id, type);
1010 return GATT_ILLEGAL_PARAMETER;
1011 }
1012
1013 if (gatt_is_clcb_allocated(conn_id)) {
1014 GATT_TRACE_ERROR("GATTC_Write GATT_BUSY conn_id = %d", conn_id);
1015 return GATT_BUSY;
1016 }
1017
1018 if ((p_clcb = gatt_clcb_alloc(conn_id)) != NULL ) {
1019 p_clcb->operation = GATTC_OPTYPE_WRITE;
1020 p_clcb->op_subtype = type;
1021 p_clcb->auth_req = p_write->auth_req;
1022
1023 if (( p_clcb->p_attr_buf = (UINT8 *)osi_malloc((UINT16)sizeof(tGATT_VALUE))) != NULL) {
1024 memcpy(p_clcb->p_attr_buf, (void *)p_write, sizeof(tGATT_VALUE));
1025
1026 p = (tGATT_VALUE *)p_clcb->p_attr_buf;
1027 if (type == GATT_WRITE_PREPARE) {
1028 p_clcb->start_offset = p_write->offset;
1029 p->offset = 0;
1030 }
1031
1032 if (gatt_security_check_start(p_clcb) == 0) {
1033 status = GATT_NO_RESOURCES;
1034 }
1035 } else {
1036 status = GATT_NO_RESOURCES;
1037 }
1038
1039 if (status == GATT_NO_RESOURCES) {
1040 gatt_clcb_dealloc(p_clcb);
1041 }
1042 } else {
1043 status = GATT_NO_RESOURCES;
1044 }
1045 return status;
1046 }
1047
1048
1049 /*******************************************************************************
1050 **
1051 ** Function GATTC_ExecuteWrite
1052 **
1053 ** Description This function is called to send an Execute write request to
1054 ** the server.
1055 **
1056 ** Parameters conn_id: connection identifier.
1057 ** is_execute - to execute or cancel the prepare write request(s)
1058 **
1059 ** Returns GATT_SUCCESS if command started successfully.
1060 **
1061 *******************************************************************************/
GATTC_ExecuteWrite(UINT16 conn_id,BOOLEAN is_execute)1062 tGATT_STATUS GATTC_ExecuteWrite (UINT16 conn_id, BOOLEAN is_execute)
1063 {
1064 tGATT_STATUS status = GATT_SUCCESS;
1065 tGATT_CLCB *p_clcb;
1066 tGATT_EXEC_FLAG flag;
1067 tGATT_IF gatt_if = GATT_GET_GATT_IF(conn_id);
1068 UINT8 tcb_idx = GATT_GET_TCB_IDX(conn_id);
1069 tGATT_TCB *p_tcb = gatt_get_tcb_by_idx(tcb_idx);
1070 tGATT_REG *p_reg = gatt_get_regcb(gatt_if);
1071
1072 GATT_TRACE_API ("GATTC_ExecuteWrite conn_id=%d is_execute=%d", conn_id, is_execute);
1073
1074 if ( (p_tcb == NULL) || (p_reg == NULL) ) {
1075 GATT_TRACE_ERROR("GATTC_ExecuteWrite Illegal param: conn_id %d", conn_id);
1076 return GATT_ILLEGAL_PARAMETER;
1077 }
1078
1079 if (gatt_is_clcb_allocated(conn_id)) {
1080 GATT_TRACE_ERROR("GATTC_Write GATT_BUSY conn_id = %d", conn_id);
1081 return GATT_BUSY;
1082 }
1083
1084 if ((p_clcb = gatt_clcb_alloc(conn_id)) != NULL) {
1085 p_clcb->operation = GATTC_OPTYPE_EXE_WRITE;
1086 flag = is_execute ? GATT_PREP_WRITE_EXEC : GATT_PREP_WRITE_CANCEL;
1087 gatt_send_queue_write_cancel (p_clcb->p_tcb, p_clcb, flag);
1088 } else {
1089 GATT_TRACE_ERROR("Unable to allocate client CB for conn_id %d ", conn_id);
1090 status = GATT_NO_RESOURCES;
1091 }
1092 return status;
1093 }
1094
1095 /*******************************************************************************
1096 **
1097 ** Function GATTC_SendHandleValueConfirm
1098 **
1099 ** Description This function is called to send a handle value confirmation
1100 ** as response to a handle value notification from server.
1101 **
1102 ** Parameters conn_id: connection identifier.
1103 ** handle: the handle of the attribute confirmation.
1104 **
1105 ** Returns GATT_SUCCESS if command started successfully.
1106 **
1107 *******************************************************************************/
GATTC_SendHandleValueConfirm(UINT16 conn_id,UINT16 handle)1108 tGATT_STATUS GATTC_SendHandleValueConfirm (UINT16 conn_id, UINT16 handle)
1109 {
1110 tGATT_STATUS ret = GATT_ILLEGAL_PARAMETER;
1111 tGATT_TCB *p_tcb = gatt_get_tcb_by_idx(GATT_GET_TCB_IDX(conn_id));
1112
1113 GATT_TRACE_API ("GATTC_SendHandleValueConfirm conn_id=%d handle=0x%x", conn_id, handle);
1114
1115 if (p_tcb) {
1116 if (p_tcb->ind_count > 0 ) {
1117 btu_stop_timer (&p_tcb->ind_ack_timer_ent);
1118
1119 GATT_TRACE_DEBUG ("notif_count=%d ", p_tcb->ind_count);
1120 /* send confirmation now */
1121 ret = attp_send_cl_msg(p_tcb, 0, GATT_HANDLE_VALUE_CONF, (tGATT_CL_MSG *)&handle);
1122
1123 p_tcb->ind_count = 0;
1124
1125 } else {
1126 GATT_TRACE_DEBUG ("GATTC_SendHandleValueConfirm - conn_id: %u - ignored not waiting for indicaiton ack", conn_id);
1127 ret = GATT_SUCCESS;
1128 }
1129 } else {
1130 GATT_TRACE_ERROR ("GATTC_SendHandleValueConfirm - Unknown conn_id: %u", conn_id);
1131 }
1132 return ret;
1133 }
1134
1135 #endif ///GATTC_INCLUDED == 1
1136
1137 /*******************************************************************************/
1138 /* */
1139 /* GATT APIs */
1140 /* */
1141 /*******************************************************************************/
1142 /*******************************************************************************
1143 **
1144 ** Function GATT_SetIdleTimeout
1145 **
1146 ** Description This function (common to both client and server) sets the idle
1147 ** timeout for a tansport connection
1148 **
1149 ** Parameter bd_addr: target device bd address.
1150 ** idle_tout: timeout value in seconds.
1151 **
1152 ** Returns void
1153 **
1154 *******************************************************************************/
GATT_SetIdleTimeout(BD_ADDR bd_addr,UINT16 idle_tout,tBT_TRANSPORT transport)1155 void GATT_SetIdleTimeout (BD_ADDR bd_addr, UINT16 idle_tout, tBT_TRANSPORT transport)
1156 {
1157 tGATT_TCB *p_tcb;
1158 BOOLEAN status = 0;
1159
1160 if ((p_tcb = gatt_find_tcb_by_addr (bd_addr, transport)) != NULL) {
1161 if (p_tcb->att_lcid == L2CAP_ATT_CID) {
1162 status = L2CA_SetFixedChannelTout (bd_addr, L2CAP_ATT_CID, idle_tout);
1163
1164 if (idle_tout == GATT_LINK_IDLE_TIMEOUT_WHEN_NO_APP) {
1165 L2CA_SetIdleTimeoutByBdAddr(p_tcb->peer_bda,
1166 GATT_LINK_IDLE_TIMEOUT_WHEN_NO_APP, BT_TRANSPORT_LE);
1167 }
1168 } else {
1169 status = L2CA_SetIdleTimeout (p_tcb->att_lcid, idle_tout, 0);
1170 }
1171 }
1172
1173 #if (CONFIG_BT_STACK_NO_LOG)
1174 (void) status;
1175 #endif
1176
1177 GATT_TRACE_API ("GATT_SetIdleTimeout idle_tout=%d status=%d(1-OK 0-not performed)",
1178 idle_tout, status);
1179 }
1180
1181
1182 /*******************************************************************************
1183 **
1184 ** Function GATT_Register
1185 **
1186 ** Description This function is called to register an application
1187 ** with GATT
1188 **
1189 ** Parameter p_app_uuid128: Application UUID
1190 ** p_cb_info: callback functions.
1191 **
1192 ** Returns 0 for error, otherwise the index of the client registered with GATT
1193 **
1194 *******************************************************************************/
GATT_Register(tBT_UUID * p_app_uuid128,const tGATT_CBACK * p_cb_info)1195 tGATT_IF GATT_Register (tBT_UUID *p_app_uuid128, const tGATT_CBACK *p_cb_info)
1196 {
1197 tGATT_REG *p_reg;
1198 UINT8 i_gatt_if = 0;
1199 tGATT_IF gatt_if = 0;
1200
1201 GATT_TRACE_API ("GATT_Register");
1202 gatt_dbg_display_uuid(*p_app_uuid128);
1203
1204 for (i_gatt_if = 0, p_reg = gatt_cb.cl_rcb; i_gatt_if < GATT_MAX_APPS; i_gatt_if++, p_reg++) {
1205 if (p_reg->in_use && !memcmp(p_app_uuid128->uu.uuid128, p_reg->app_uuid128.uu.uuid128, LEN_UUID_128)) {
1206 GATT_TRACE_ERROR("application already registered.");
1207 return 0;
1208 }
1209 }
1210
1211 for (i_gatt_if = 0, p_reg = gatt_cb.cl_rcb; i_gatt_if < GATT_MAX_APPS; i_gatt_if++, p_reg++) {
1212 if (!p_reg->in_use) {
1213 memset(p_reg, 0 , sizeof(tGATT_REG));
1214 i_gatt_if++; /* one based number */
1215 p_reg->app_uuid128 = *p_app_uuid128;
1216 gatt_if =
1217 p_reg->gatt_if = (tGATT_IF)i_gatt_if;
1218 p_reg->app_cb = *p_cb_info;
1219 p_reg->in_use = 1;
1220
1221 break;
1222 }
1223 }
1224 GATT_TRACE_API ("allocated gatt_if=%d\n", gatt_if);
1225 return gatt_if;
1226 }
1227
1228
1229 /*******************************************************************************
1230 **
1231 ** Function GATT_Deregister
1232 **
1233 ** Description This function deregistered the application from GATT.
1234 **
1235 ** Parameters gatt_if: application interface.
1236 **
1237 ** Returns None.
1238 **
1239 *******************************************************************************/
GATT_Deregister(tGATT_IF gatt_if)1240 void GATT_Deregister (tGATT_IF gatt_if)
1241 {
1242 tGATT_REG *p_reg = gatt_get_regcb(gatt_if);
1243 tGATT_TCB *p_tcb;
1244 tGATT_CLCB *p_clcb;
1245 list_node_t *p_node = NULL;
1246 list_node_t *p_next = NULL;
1247 #if (GATTS_INCLUDED == 1)
1248 UINT8 ii;
1249 tGATT_SR_REG *p_sreg;
1250 #endif ///GATTS_INCLUDED == 1
1251 GATT_TRACE_API ("GATT_Deregister gatt_if=%d", gatt_if);
1252 /* Index 0 is GAP and is never deregistered */
1253 if ( (gatt_if == 0) || (p_reg == NULL) ) {
1254 GATT_TRACE_ERROR ("GATT_Deregister with invalid gatt_if: %u", gatt_if);
1255 return;
1256 }
1257
1258 /* stop all services */
1259 /* todo an applcaiton can not be deregistered if its services is also used by other application
1260 deregisteration need to bed performed in an orderly fashion
1261 no check for now */
1262 #if (GATTS_INCLUDED == 1)
1263 for (ii = 0, p_sreg = gatt_cb.sr_reg; ii < GATT_MAX_SR_PROFILES; ii++, p_sreg++) {
1264 if (p_sreg->in_use && (p_sreg->gatt_if == gatt_if)) {
1265 GATTS_StopService(p_sreg->s_hdl);
1266 }
1267 }
1268 /* free all services db buffers if owned by this application */
1269 gatt_free_srvc_db_buffer_app_id(&p_reg->app_uuid128);
1270 #endif ///GATTS_INCLUDED == 1
1271 /* When an application deregisters, check remove the link associated with the app */
1272
1273 for(p_node = list_begin(gatt_cb.p_tcb_list); p_node; p_node = p_next) {
1274 p_tcb = list_node(p_node);
1275 p_next = list_next(p_node);
1276 if (p_tcb->in_use) {
1277 if (gatt_get_ch_state(p_tcb) != GATT_CH_CLOSE) {
1278 gatt_update_app_use_link_flag(gatt_if, p_tcb, 0, 0);
1279 if (!gatt_num_apps_hold_link(p_tcb)) {
1280 /* this will disconnect the link or cancel the pending connect request at lower layer*/
1281 gatt_disconnect(p_tcb);
1282 }
1283 }
1284
1285 list_node_t *p_node_clcb = NULL;
1286 list_node_t *p_node_next = NULL;
1287 for(p_node_clcb = list_begin(gatt_cb.p_clcb_list); p_node_clcb; p_node_clcb = p_node_next) {
1288 p_clcb = list_node(p_node_clcb);
1289 p_node_next = list_next(p_node_clcb);
1290 if (p_clcb->in_use &&
1291 (p_clcb->p_reg->gatt_if == gatt_if) &&
1292 (p_clcb->p_tcb->tcb_idx == p_tcb->tcb_idx)) {
1293 btu_stop_timer(&p_clcb->rsp_timer_ent);
1294 gatt_clcb_dealloc (p_clcb);
1295 break;
1296 }
1297 }
1298 }
1299 }
1300
1301 gatt_deregister_bgdev_list(gatt_if);
1302 /* update the listen mode */
1303 #if (defined(BLE_PERIPHERAL_MODE_SUPPORT) && (BLE_PERIPHERAL_MODE_SUPPORT == 1))
1304 GATT_Listen(gatt_if, 0, NULL);
1305 #endif
1306
1307 memset (p_reg, 0, sizeof(tGATT_REG));
1308 }
1309
1310
1311 /*******************************************************************************
1312 **
1313 ** Function GATT_StartIf
1314 **
1315 ** Description This function is called after registration to start receiving
1316 ** callbacks for registered interface. Function may call back
1317 ** with connection status and queued notifications
1318 **
1319 ** Parameter gatt_if: application interface.
1320 **
1321 ** Returns None.
1322 **
1323 *******************************************************************************/
GATT_StartIf(tGATT_IF gatt_if)1324 void GATT_StartIf (tGATT_IF gatt_if)
1325 {
1326 tGATT_REG *p_reg;
1327 tGATT_TCB *p_tcb;
1328 BD_ADDR bda;
1329 UINT8 start_idx, found_idx;
1330 UINT16 conn_id;
1331 tGATT_TRANSPORT transport ;
1332
1333 GATT_TRACE_API ("GATT_StartIf gatt_if=%d", gatt_if);
1334 if ((p_reg = gatt_get_regcb(gatt_if)) != NULL) {
1335 start_idx = 0;
1336 while (gatt_find_the_connected_bda(start_idx, bda, &found_idx, &transport)) {
1337 p_tcb = gatt_find_tcb_by_addr(bda, transport);
1338 if (p_reg->app_cb.p_conn_cb && p_tcb) {
1339 conn_id = GATT_CREATE_CONN_ID(p_tcb->tcb_idx, gatt_if);
1340 (*p_reg->app_cb.p_conn_cb)(gatt_if, bda, conn_id, 1, 0, transport);
1341 }
1342 start_idx = ++found_idx;
1343 }
1344 }
1345 }
1346
1347
1348 /*******************************************************************************
1349 **
1350 ** Function GATT_Connect
1351 **
1352 ** Description This function initiate a connection to a remote device on GATT
1353 ** channel.
1354 **
1355 ** Parameters gatt_if: application interface
1356 ** bd_addr: peer device address.
1357 ** bd_addr_type: peer device address type.
1358 ** is_direct: is a direct connection or a background auto connection
1359 **
1360 ** Returns 1 if connection started; 0 if connection start failure.
1361 **
1362 *******************************************************************************/
GATT_Connect(tGATT_IF gatt_if,BD_ADDR bd_addr,tBLE_ADDR_TYPE bd_addr_type,BOOLEAN is_direct,tBT_TRANSPORT transport,BOOLEAN is_aux)1363 BOOLEAN GATT_Connect (tGATT_IF gatt_if, BD_ADDR bd_addr, tBLE_ADDR_TYPE bd_addr_type,
1364 BOOLEAN is_direct, tBT_TRANSPORT transport, BOOLEAN is_aux)
1365 {
1366 tGATT_REG *p_reg;
1367 BOOLEAN status = 0;
1368
1369 GATT_TRACE_API ("GATT_Connect gatt_if=%d", gatt_if);
1370
1371 /* Make sure app is registered */
1372 if ((p_reg = gatt_get_regcb(gatt_if)) == NULL) {
1373 GATT_TRACE_ERROR("GATT_Connect - gatt_if =%d is not registered", gatt_if);
1374 return (0);
1375 }
1376
1377 if (is_direct) {
1378 status = gatt_act_connect (p_reg, bd_addr, bd_addr_type, transport, is_aux);
1379 } else {
1380 if (transport == BT_TRANSPORT_LE) {
1381 status = gatt_update_auto_connect_dev(gatt_if, 1, bd_addr, 1);
1382 } else {
1383 GATT_TRACE_ERROR("Unsupported transport for background connection");
1384 }
1385 }
1386
1387 return status;
1388
1389 }
1390
1391 /*******************************************************************************
1392 **
1393 ** Function GATT_CancelConnect
1394 **
1395 ** Description This function terminate the connection initaition to a remote
1396 ** device on GATT channel.
1397 **
1398 ** Parameters gatt_if: client interface. If 0 used as unconditionally disconnect,
1399 ** typically used for direct connection cancellation.
1400 ** bd_addr: peer device address.
1401 **
1402 ** Returns 1 if connection started; 0 if connection start failure.
1403 **
1404 *******************************************************************************/
GATT_CancelConnect(tGATT_IF gatt_if,BD_ADDR bd_addr,BOOLEAN is_direct)1405 BOOLEAN GATT_CancelConnect (tGATT_IF gatt_if, BD_ADDR bd_addr, BOOLEAN is_direct)
1406 {
1407 tGATT_REG *p_reg;
1408 tGATT_TCB *p_tcb;
1409 BOOLEAN status = 1;
1410 tGATT_IF temp_gatt_if;
1411 UINT8 start_idx, found_idx;
1412
1413 GATT_TRACE_API ("GATT_CancelConnect gatt_if=%d", gatt_if);
1414
1415 if ((gatt_if != 0) && ((p_reg = gatt_get_regcb(gatt_if)) == NULL)) {
1416 GATT_TRACE_ERROR("GATT_CancelConnect - gatt_if =%d is not registered", gatt_if);
1417 return (0);
1418 }
1419
1420 if (is_direct) {
1421 if (!gatt_if) {
1422 GATT_TRACE_DEBUG("GATT_CancelConnect - unconditional");
1423 start_idx = 0;
1424 /* only LE connection can be cancelled */
1425 p_tcb = gatt_find_tcb_by_addr(bd_addr, BT_TRANSPORT_LE);
1426 if (p_tcb && gatt_num_apps_hold_link(p_tcb)) {
1427 while (status && gatt_find_app_hold_link(p_tcb, start_idx, &found_idx, &temp_gatt_if)) {
1428 status = gatt_cancel_open(temp_gatt_if, bd_addr);
1429 start_idx = ++found_idx;
1430 }
1431 } else {
1432 GATT_TRACE_ERROR("GATT_CancelConnect - no app found");
1433 status = 0;
1434 }
1435 } else {
1436 status = gatt_cancel_open(gatt_if, bd_addr);
1437 }
1438 } else {
1439 if (!gatt_if) {
1440 if (gatt_get_num_apps_for_bg_dev(bd_addr)) {
1441 while (gatt_find_app_for_bg_dev(bd_addr, &temp_gatt_if)) {
1442 gatt_remove_bg_dev_for_app(temp_gatt_if, bd_addr);
1443 }
1444 } else {
1445 GATT_TRACE_ERROR("GATT_CancelConnect -no app associated with the bg device for unconditional removal");
1446 status = 0;
1447 }
1448 } else {
1449 status = gatt_remove_bg_dev_for_app(gatt_if, bd_addr);
1450 }
1451 }
1452
1453 return status;
1454 }
1455
1456 /*******************************************************************************
1457 **
1458 ** Function GATT_Disconnect
1459 **
1460 ** Description This function disconnect the GATT channel for this registered
1461 ** application.
1462 **
1463 ** Parameters conn_id: connection identifier.
1464 **
1465 ** Returns GATT_SUCCESS if disconnected.
1466 **
1467 *******************************************************************************/
GATT_Disconnect(UINT16 conn_id)1468 tGATT_STATUS GATT_Disconnect (UINT16 conn_id)
1469 {
1470 tGATT_STATUS ret = GATT_ILLEGAL_PARAMETER;
1471 tGATT_TCB *p_tcb = NULL;
1472 tGATT_IF gatt_if = GATT_GET_GATT_IF(conn_id);
1473 UINT8 tcb_idx = GATT_GET_TCB_IDX(conn_id);
1474
1475 GATT_TRACE_API ("GATT_Disconnect conn_id=%d ", conn_id);
1476
1477 p_tcb = gatt_get_tcb_by_idx(tcb_idx);
1478
1479 if (p_tcb) {
1480 gatt_update_app_use_link_flag(gatt_if, p_tcb, 0, 0);
1481 if (!gatt_num_apps_hold_link(p_tcb)) {
1482 gatt_disconnect(p_tcb);
1483 }
1484 ret = GATT_SUCCESS;
1485 }
1486 return ret;
1487 }
1488
1489 /*******************************************************************************
1490 **
1491 ** Function GATT_SendServiceChangeIndication
1492 **
1493 ** Description This function is to send a service change indication
1494 **
1495 ** Parameters bd_addr: peer device address.
1496 **
1497 ** Returns None.
1498 **
1499 *******************************************************************************/
GATT_SendServiceChangeIndication(BD_ADDR bd_addr)1500 tGATT_STATUS GATT_SendServiceChangeIndication (BD_ADDR bd_addr)
1501 {
1502 UINT8 start_idx, found_idx;
1503 BOOLEAN srv_chg_ind_pending = 0;
1504 tGATT_TCB *p_tcb;
1505 tBT_TRANSPORT transport;
1506 tGATT_STATUS status = GATT_NOT_FOUND;
1507 if (GATTS_SEND_SERVICE_CHANGE_MODE == GATTS_SEND_SERVICE_CHANGE_AUTO) {
1508 status = GATT_WRONG_STATE;
1509 GATT_TRACE_ERROR ("%s can't send service change indication manually, please configure the option through menuconfig", __func__);
1510 return status;
1511 }
1512
1513 if(bd_addr) {
1514 status = gatt_send_srv_chg_ind(bd_addr);
1515 } else {
1516 start_idx = 0;
1517 BD_ADDR addr;
1518 while (gatt_find_the_connected_bda(start_idx, addr, &found_idx, &transport)) {
1519 p_tcb = gatt_get_tcb_by_idx(found_idx);
1520 srv_chg_ind_pending = gatt_is_srv_chg_ind_pending(p_tcb);
1521
1522 if (!srv_chg_ind_pending) {
1523 status = gatt_send_srv_chg_ind(addr);
1524 } else {
1525 status = GATT_BUSY;
1526 GATT_TRACE_DEBUG("discard srv chg - already has one in the queue");
1527 }
1528 start_idx = ++found_idx;
1529 }
1530 }
1531
1532 return status;
1533 }
1534
1535 /*******************************************************************************
1536 **
1537 ** Function GATT_GetConnectionInfor
1538 **
1539 ** Description This function use conn_id to find its associated BD address and applciation
1540 ** interface
1541 **
1542 ** Parameters conn_id: connection id (input)
1543 ** p_gatt_if: application interface (output)
1544 ** bd_addr: peer device address. (output)
1545 **
1546 ** Returns 1 the logical link information is found for conn_id
1547 **
1548 *******************************************************************************/
GATT_GetConnectionInfor(UINT16 conn_id,tGATT_IF * p_gatt_if,BD_ADDR bd_addr,tBT_TRANSPORT * p_transport)1549 BOOLEAN GATT_GetConnectionInfor(UINT16 conn_id, tGATT_IF *p_gatt_if, BD_ADDR bd_addr,
1550 tBT_TRANSPORT *p_transport)
1551 {
1552
1553 tGATT_IF gatt_if = GATT_GET_GATT_IF(conn_id);
1554 tGATT_REG *p_reg = gatt_get_regcb(gatt_if);
1555 UINT8 tcb_idx = GATT_GET_TCB_IDX(conn_id);
1556 tGATT_TCB *p_tcb = gatt_get_tcb_by_idx(tcb_idx);
1557 BOOLEAN status = 0;
1558
1559 GATT_TRACE_API ("GATT_GetConnectionInfor conn_id=%d", conn_id);
1560
1561 if (p_tcb && p_reg ) {
1562 memcpy(bd_addr, p_tcb->peer_bda, BD_ADDR_LEN);
1563 *p_gatt_if = gatt_if;
1564 *p_transport = p_tcb->transport;
1565 status = 1;
1566 }
1567 return status;
1568 }
1569
1570
1571 /*******************************************************************************
1572 **
1573 ** Function GATT_GetConnIdIfConnected
1574 **
1575 ** Description This function find the conn_id if the logical link for BD address
1576 ** and applciation interface is connected
1577 **
1578 ** Parameters gatt_if: application interface (input)
1579 ** bd_addr: peer device address. (input)
1580 ** p_conn_id: connection id (output)
1581 ** transport: transport option
1582 **
1583 ** Returns 1 the logical link is connected
1584 **
1585 *******************************************************************************/
GATT_GetConnIdIfConnected(tGATT_IF gatt_if,BD_ADDR bd_addr,UINT16 * p_conn_id,tBT_TRANSPORT transport)1586 BOOLEAN GATT_GetConnIdIfConnected(tGATT_IF gatt_if, BD_ADDR bd_addr, UINT16 *p_conn_id,
1587 tBT_TRANSPORT transport)
1588 {
1589 tGATT_REG *p_reg = gatt_get_regcb(gatt_if);
1590 tGATT_TCB *p_tcb = gatt_find_tcb_by_addr(bd_addr, transport);
1591 BOOLEAN status = 0;
1592
1593 if (p_reg && p_tcb && (gatt_get_ch_state(p_tcb) == GATT_CH_OPEN) ) {
1594 *p_conn_id = GATT_CREATE_CONN_ID(p_tcb->tcb_idx, gatt_if);
1595 status = 1;
1596 }
1597
1598 GATT_TRACE_API ("GATT_GetConnIdIfConnected status=%d\n", status);
1599 return status;
1600 }
1601
1602
1603 /*******************************************************************************
1604 **
1605 ** Function GATT_Listen
1606 **
1607 ** Description This function start or stop LE advertisement and listen for
1608 ** connection.
1609 **
1610 ** Parameters gatt_if: application interface
1611 ** p_bd_addr: listen for specific address connection, or NULL for
1612 ** listen to all device connection.
1613 ** start: start or stop listening.
1614 **
1615 ** Returns 1 if advertisement is started; 0 if adv start failure.
1616 **
1617 *******************************************************************************/
GATT_Listen(tGATT_IF gatt_if,BOOLEAN start,BD_ADDR_PTR bd_addr)1618 BOOLEAN GATT_Listen (tGATT_IF gatt_if, BOOLEAN start, BD_ADDR_PTR bd_addr)
1619 {
1620 tGATT_REG *p_reg;
1621
1622 GATT_TRACE_API ("GATT_Listen gatt_if=%d", gatt_if);
1623
1624 /* Make sure app is registered */
1625 if ((p_reg = gatt_get_regcb(gatt_if)) == NULL) {
1626 GATT_TRACE_ERROR("GATT_Listen - gatt_if =%d is not registered", gatt_if);
1627 return (0);
1628 }
1629
1630 if (bd_addr != NULL) {
1631 gatt_update_auto_connect_dev(gatt_if, start, bd_addr, 0);
1632 } else {
1633 p_reg->listening = start ? GATT_LISTEN_TO_ALL : GATT_LISTEN_TO_NONE;
1634 }
1635
1636 return gatt_update_listen_mode();
1637 }
1638
1639 #endif
1640