1 /******************************************************************************
2 *
3 * Copyright (C) 2008-2014 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 ATT protocol functions
22 *
23 ******************************************************************************/
24
25 #include "common/bt_target.h"
26 #include "osi/allocator.h"
27
28 #if BLE_INCLUDED == 1
29
30 #include "gatt_int.h"
31 #include "stack/l2c_api.h"
32
33 #define GATT_HDR_FIND_TYPE_VALUE_LEN 21
34 #define GATT_OP_CODE_SIZE 1
35 /**********************************************************************
36 ** ATT protocl message building utility *
37 ***********************************************************************/
38 /*******************************************************************************
39 **
40 ** Function attp_build_mtu_exec_cmd
41 **
42 ** Description Build a exchange MTU request
43 **
44 ** Returns None.
45 **
46 *******************************************************************************/
attp_build_mtu_cmd(UINT8 op_code,UINT16 rx_mtu)47 BT_HDR *attp_build_mtu_cmd(UINT8 op_code, UINT16 rx_mtu)
48 {
49 BT_HDR *p_buf = NULL;
50 UINT8 *p;
51
52 if ((p_buf = (BT_HDR *)osi_malloc(sizeof(BT_HDR) + GATT_HDR_SIZE + L2CAP_MIN_OFFSET)) != NULL) {
53 p = (UINT8 *)(p_buf + 1) + L2CAP_MIN_OFFSET;
54
55 UINT8_TO_STREAM (p, op_code);
56 UINT16_TO_STREAM (p, rx_mtu);
57
58 p_buf->offset = L2CAP_MIN_OFFSET;
59 p_buf->len = GATT_HDR_SIZE; /* opcode + 2 bytes mtu */
60 }
61 return p_buf;
62 }
63 /*******************************************************************************
64 **
65 ** Function attp_build_exec_write_cmd
66 **
67 ** Description Build a execute write request or response.
68 **
69 ** Returns None.
70 **
71 *******************************************************************************/
attp_build_exec_write_cmd(UINT8 op_code,UINT8 flag)72 BT_HDR *attp_build_exec_write_cmd (UINT8 op_code, UINT8 flag)
73 {
74 BT_HDR *p_buf = NULL;
75 UINT8 *p;
76
77 if ((p_buf = (BT_HDR *)osi_malloc(GATT_DATA_BUF_SIZE)) != NULL) {
78 p = (UINT8 *)(p_buf + 1) + L2CAP_MIN_OFFSET;
79
80 p_buf->offset = L2CAP_MIN_OFFSET;
81 p_buf->len = GATT_OP_CODE_SIZE;
82
83 UINT8_TO_STREAM (p, op_code);
84
85 if (op_code == GATT_REQ_EXEC_WRITE) {
86 flag &= GATT_PREP_WRITE_EXEC;
87 UINT8_TO_STREAM (p, flag);
88 p_buf->len += 1;
89 }
90
91 }
92
93 return p_buf;
94 }
95
96 /*******************************************************************************
97 **
98 ** Function attp_build_err_cmd
99 **
100 ** Description Build a exchange MTU request
101 **
102 ** Returns None.
103 **
104 *******************************************************************************/
attp_build_err_cmd(UINT8 cmd_code,UINT16 err_handle,UINT8 reason)105 BT_HDR *attp_build_err_cmd(UINT8 cmd_code, UINT16 err_handle, UINT8 reason)
106 {
107 BT_HDR *p_buf = NULL;
108 UINT8 *p;
109
110 if ((p_buf = (BT_HDR *)osi_malloc(sizeof(BT_HDR) + L2CAP_MIN_OFFSET + 5)) != NULL) {
111 p = (UINT8 *)(p_buf + 1) + L2CAP_MIN_OFFSET;
112
113 UINT8_TO_STREAM (p, GATT_RSP_ERROR);
114 UINT8_TO_STREAM (p, cmd_code);
115 UINT16_TO_STREAM(p, err_handle);
116 UINT8_TO_STREAM (p, reason);
117
118 p_buf->offset = L2CAP_MIN_OFFSET;
119 /* GATT_HDR_SIZE (1B ERR_RSP op code+ 2B handle) + 1B cmd_op_code + 1B status */
120 p_buf->len = GATT_HDR_SIZE + 1 + 1;
121 }
122 return p_buf;
123 }
124 /*******************************************************************************
125 **
126 ** Function attp_build_browse_cmd
127 **
128 ** Description Build a read information request or read by type request
129 **
130 ** Returns None.
131 **
132 *******************************************************************************/
attp_build_browse_cmd(UINT8 op_code,UINT16 s_hdl,UINT16 e_hdl,tBT_UUID uuid)133 BT_HDR *attp_build_browse_cmd(UINT8 op_code, UINT16 s_hdl, UINT16 e_hdl, tBT_UUID uuid)
134 {
135 BT_HDR *p_buf = NULL;
136 UINT8 *p;
137
138 if ((p_buf = (BT_HDR *)osi_malloc(sizeof(BT_HDR) + 8 + L2CAP_MIN_OFFSET)) != NULL) {
139 p = (UINT8 *)(p_buf + 1) + L2CAP_MIN_OFFSET;
140 /* Describe the built message location and size */
141 p_buf->offset = L2CAP_MIN_OFFSET;
142 p_buf->len = GATT_OP_CODE_SIZE + 4;
143
144 UINT8_TO_STREAM (p, op_code);
145 UINT16_TO_STREAM (p, s_hdl);
146 UINT16_TO_STREAM (p, e_hdl);
147 p_buf->len += gatt_build_uuid_to_stream(&p, uuid);
148 }
149
150 return p_buf;
151 }
152 /*******************************************************************************
153 **
154 ** Function attp_build_read_handles_cmd
155 **
156 ** Description Build a read by type and value request.
157 **
158 ** Returns pointer to the command buffer.
159 **
160 *******************************************************************************/
attp_build_read_by_type_value_cmd(UINT16 payload_size,tGATT_FIND_TYPE_VALUE * p_value_type)161 BT_HDR *attp_build_read_by_type_value_cmd (UINT16 payload_size, tGATT_FIND_TYPE_VALUE *p_value_type)
162 {
163 BT_HDR *p_buf = NULL;
164 UINT8 *p;
165 UINT16 len = p_value_type->value_len;
166
167 if ((p_buf = (BT_HDR *)osi_malloc((UINT16)(sizeof(BT_HDR) + payload_size + L2CAP_MIN_OFFSET))) != NULL) {
168 p = (UINT8 *)(p_buf + 1) + L2CAP_MIN_OFFSET;
169
170 p_buf->offset = L2CAP_MIN_OFFSET;
171 p_buf->len = 5; /* opcode + s_handle + e_handle */
172
173 UINT8_TO_STREAM (p, GATT_REQ_FIND_TYPE_VALUE);
174 UINT16_TO_STREAM (p, p_value_type->s_handle);
175 UINT16_TO_STREAM (p, p_value_type->e_handle);
176
177 p_buf->len += gatt_build_uuid_to_stream(&p, p_value_type->uuid);
178
179 if (p_value_type->value_len + p_buf->len > payload_size ) {
180 len = payload_size - p_buf->len;
181 }
182
183 memcpy (p, p_value_type->value, len);
184 p_buf->len += len;
185 }
186
187 return p_buf;
188 }
189 /*******************************************************************************
190 **
191 ** Function attp_build_read_multi_cmd
192 **
193 ** Description Build a read multiple request
194 **
195 ** Returns None.
196 **
197 *******************************************************************************/
attp_build_read_multi_cmd(UINT16 payload_size,UINT16 num_handle,UINT16 * p_handle)198 BT_HDR *attp_build_read_multi_cmd(UINT16 payload_size, UINT16 num_handle, UINT16 *p_handle)
199 {
200 BT_HDR *p_buf = NULL;
201 UINT8 *p, i = 0;
202
203 if ((p_buf = (BT_HDR *)osi_malloc((UINT16)(sizeof(BT_HDR) + num_handle * 2 + 1 + L2CAP_MIN_OFFSET))) != NULL) {
204 p = (UINT8 *)(p_buf + 1) + L2CAP_MIN_OFFSET;
205
206 p_buf->offset = L2CAP_MIN_OFFSET;
207 p_buf->len = 1;
208
209 UINT8_TO_STREAM (p, GATT_REQ_READ_MULTI);
210
211 for (i = 0; i < num_handle && p_buf->len + 2 <= payload_size; i ++) {
212 UINT16_TO_STREAM (p, *(p_handle + i));
213 p_buf->len += 2;
214 }
215 }
216
217 return p_buf;
218 }
219 /*******************************************************************************
220 **
221 ** Function attp_build_handle_cmd
222 **
223 ** Description Build a read /read blob request
224 **
225 ** Returns None.
226 **
227 *******************************************************************************/
attp_build_handle_cmd(UINT8 op_code,UINT16 handle,UINT16 offset)228 BT_HDR *attp_build_handle_cmd(UINT8 op_code, UINT16 handle, UINT16 offset)
229 {
230 BT_HDR *p_buf = NULL;
231 UINT8 *p;
232
233 if ((p_buf = (BT_HDR *)osi_malloc(sizeof(BT_HDR) + 5 + L2CAP_MIN_OFFSET)) != NULL) {
234 p = (UINT8 *)(p_buf + 1) + L2CAP_MIN_OFFSET;
235
236 p_buf->offset = L2CAP_MIN_OFFSET;
237
238 UINT8_TO_STREAM (p, op_code);
239 p_buf->len = 1;
240
241 UINT16_TO_STREAM (p, handle);
242 p_buf->len += 2;
243
244 if (op_code == GATT_REQ_READ_BLOB) {
245 UINT16_TO_STREAM (p, offset);
246 p_buf->len += 2;
247 }
248
249 }
250
251 return p_buf;
252 }
253 /*******************************************************************************
254 **
255 ** Function attp_build_opcode_cmd
256 **
257 ** Description Build a request/response with opcode only.
258 **
259 ** Returns None.
260 **
261 *******************************************************************************/
attp_build_opcode_cmd(UINT8 op_code)262 BT_HDR *attp_build_opcode_cmd(UINT8 op_code)
263 {
264 BT_HDR *p_buf = NULL;
265 UINT8 *p;
266
267 if ((p_buf = (BT_HDR *)osi_malloc(sizeof(BT_HDR) + 1 + L2CAP_MIN_OFFSET)) != NULL) {
268 p = (UINT8 *)(p_buf + 1) + L2CAP_MIN_OFFSET;
269 p_buf->offset = L2CAP_MIN_OFFSET;
270
271 UINT8_TO_STREAM (p, op_code);
272 p_buf->len = 1;
273 }
274
275 return p_buf;
276 }
277 /*******************************************************************************
278 **
279 ** Function attp_build_value_cmd
280 **
281 ** Description Build a attribute value request
282 **
283 ** Returns None.
284 **
285 *******************************************************************************/
attp_build_value_cmd(UINT16 payload_size,UINT8 op_code,UINT16 handle,UINT16 offset,UINT16 len,UINT8 * p_data)286 BT_HDR *attp_build_value_cmd (UINT16 payload_size, UINT8 op_code, UINT16 handle,
287 UINT16 offset, UINT16 len, UINT8 *p_data)
288 {
289 BT_HDR *p_buf = NULL;
290 UINT8 *p, *pp, pair_len, *p_pair_len;
291
292 if ((p_buf = (BT_HDR *)osi_malloc((UINT16)(sizeof(BT_HDR) + payload_size + L2CAP_MIN_OFFSET))) != NULL) {
293 p = pp = (UINT8 *)(p_buf + 1) + L2CAP_MIN_OFFSET;
294
295 UINT8_TO_STREAM (p, op_code);
296 p_buf->offset = L2CAP_MIN_OFFSET;
297 p_buf->len = 1;
298
299 if (op_code == GATT_RSP_READ_BY_TYPE) {
300 p_pair_len = p;
301 pair_len = len + 2;
302 UINT8_TO_STREAM (p, pair_len);
303 p_buf->len += 1;
304 }
305 if (op_code != GATT_RSP_READ_BLOB && op_code != GATT_RSP_READ) {
306 UINT16_TO_STREAM (p, handle);
307 p_buf->len += 2;
308 }
309
310 if (op_code == GATT_REQ_PREPARE_WRITE || op_code == GATT_RSP_PREPARE_WRITE ) {
311 UINT16_TO_STREAM (p, offset);
312 p_buf->len += 2;
313 }
314
315 if (len > 0 && p_data != NULL) {
316 /* ensure data not exceed MTU size */
317 if (payload_size - p_buf->len < len) {
318 len = payload_size - p_buf->len;
319 /* update handle value pair length */
320 if (op_code == GATT_RSP_READ_BY_TYPE) {
321 *p_pair_len = (len + 2);
322 }
323
324 GATT_TRACE_WARNING("attribute value too long, to be truncated to %d", len);
325 }
326
327 ARRAY_TO_STREAM (p, p_data, len);
328 p_buf->len += len;
329 }
330 }
331 return p_buf;
332 }
333
334 /*******************************************************************************
335 **
336 ** Function attp_send_msg_to_l2cap
337 **
338 ** Description Send message to L2CAP.
339 **
340 *******************************************************************************/
attp_send_msg_to_l2cap(tGATT_TCB * p_tcb,BT_HDR * p_toL2CAP)341 tGATT_STATUS attp_send_msg_to_l2cap(tGATT_TCB *p_tcb, BT_HDR *p_toL2CAP)
342 {
343 UINT16 l2cap_ret;
344
345
346 if (p_tcb->att_lcid == L2CAP_ATT_CID) {
347 l2cap_ret = L2CA_SendFixedChnlData (L2CAP_ATT_CID, p_tcb->peer_bda, p_toL2CAP);
348 } else {
349 #if (CLASSIC_BT_INCLUDED == 1)
350 l2cap_ret = (UINT16) L2CA_DataWrite (p_tcb->att_lcid, p_toL2CAP);
351 #else
352 l2cap_ret = L2CAP_DW_FAILED;
353 #endif ///CLASSIC_BT_INCLUDED == 1
354 }
355
356 if (l2cap_ret == L2CAP_DW_FAILED) {
357 GATT_TRACE_DEBUG("ATT failed to pass msg:0x%0x to L2CAP",
358 *((UINT8 *)(p_toL2CAP + 1) + p_toL2CAP->offset));
359 return GATT_INTERNAL_ERROR;
360 } else if (l2cap_ret == L2CAP_DW_CONGESTED) {
361 GATT_TRACE_DEBUG("ATT congested, message accepted");
362 return GATT_CONGESTED;
363 }
364 return GATT_SUCCESS;
365 }
366
367 /*******************************************************************************
368 **
369 ** Function attp_build_sr_msg
370 **
371 ** Description Build ATT Server PDUs.
372 **
373 *******************************************************************************/
attp_build_sr_msg(tGATT_TCB * p_tcb,UINT8 op_code,tGATT_SR_MSG * p_msg)374 BT_HDR *attp_build_sr_msg(tGATT_TCB *p_tcb, UINT8 op_code, tGATT_SR_MSG *p_msg)
375 {
376 BT_HDR *p_cmd = NULL;
377 UINT16 offset = 0;
378
379 switch (op_code) {
380 case GATT_RSP_READ_BLOB:
381 case GATT_RSP_PREPARE_WRITE:
382 case GATT_RSP_READ_BY_TYPE:
383 case GATT_RSP_READ:
384 case GATT_HANDLE_VALUE_NOTIF:
385 case GATT_HANDLE_VALUE_IND:
386 case GATT_RSP_ERROR:
387 case GATT_RSP_MTU:
388 /* Need to check the validation of parameter p_msg*/
389 if (p_msg == NULL) {
390 GATT_TRACE_ERROR("Invalid parameters in %s, op_code=0x%x, the p_msg should not be NULL.", __func__, op_code);
391 return NULL;
392 }
393 break;
394
395 default:
396 break;
397 }
398
399 switch (op_code) {
400 case GATT_RSP_READ_BLOB:
401 case GATT_RSP_PREPARE_WRITE:
402 GATT_TRACE_EVENT ("ATT_RSP_READ_BLOB/GATT_RSP_PREPARE_WRITE: len = %d offset = %d",
403 p_msg->attr_value.len, p_msg->attr_value.offset);
404 offset = p_msg->attr_value.offset;
405 /* Coverity: [0-POSITIVE error] intended fall through */
406 /* Missing break statement between cases in switch statement */
407 /* fall through */
408 case GATT_RSP_READ_BY_TYPE:
409 case GATT_RSP_READ:
410 case GATT_HANDLE_VALUE_NOTIF:
411 case GATT_HANDLE_VALUE_IND:
412 p_cmd = attp_build_value_cmd(p_tcb->payload_size,
413 op_code,
414 p_msg->attr_value.handle,
415 offset,
416 p_msg->attr_value.len,
417 p_msg->attr_value.value);
418 break;
419
420 case GATT_RSP_WRITE:
421 p_cmd = attp_build_opcode_cmd(op_code);
422 break;
423
424 case GATT_RSP_ERROR:
425 p_cmd = attp_build_err_cmd(p_msg->error.cmd_code, p_msg->error.handle, p_msg->error.reason);
426 break;
427
428 case GATT_RSP_EXEC_WRITE:
429 p_cmd = attp_build_exec_write_cmd(op_code, 0);
430 break;
431
432 case GATT_RSP_MTU:
433 p_cmd = attp_build_mtu_cmd(op_code, p_msg->mtu);
434 break;
435
436 default:
437 GATT_TRACE_DEBUG("attp_build_sr_msg: unknown op code = %d", op_code);
438 break;
439 }
440
441 if (!p_cmd) {
442 GATT_TRACE_ERROR("No resources");
443 }
444
445 return p_cmd;
446 }
447
448 /*******************************************************************************
449 **
450 ** Function attp_send_sr_msg
451 **
452 ** Description This function sends the server response or indication message
453 ** to client.
454 **
455 ** Parameter p_tcb: pointer to the connecton control block.
456 ** p_msg: pointer to message parameters structure.
457 **
458 ** Returns GATT_SUCCESS if successfully sent; otherwise error code.
459 **
460 **
461 *******************************************************************************/
attp_send_sr_msg(tGATT_TCB * p_tcb,BT_HDR * p_msg)462 tGATT_STATUS attp_send_sr_msg (tGATT_TCB *p_tcb, BT_HDR *p_msg)
463 {
464 tGATT_STATUS cmd_sent = GATT_NO_RESOURCES;
465
466 if (p_tcb != NULL) {
467 if (p_msg != NULL) {
468 p_msg->offset = L2CAP_MIN_OFFSET;
469 cmd_sent = attp_send_msg_to_l2cap (p_tcb, p_msg);
470 }
471 }
472 return cmd_sent;
473 }
474
475 /*******************************************************************************
476 **
477 ** Function attp_cl_send_cmd
478 **
479 ** Description Send a ATT command or enqueue it.
480 **
481 ** Returns GATT_SUCCESS if command sent
482 ** GATT_CONGESTED if command sent but channel congested
483 ** GATT_CMD_STARTED if command queue up in GATT
484 ** GATT_ERROR if command sending failure
485 **
486 *******************************************************************************/
attp_cl_send_cmd(tGATT_TCB * p_tcb,UINT16 clcb_idx,UINT8 cmd_code,BT_HDR * p_cmd)487 tGATT_STATUS attp_cl_send_cmd(tGATT_TCB *p_tcb, UINT16 clcb_idx, UINT8 cmd_code, BT_HDR *p_cmd)
488 {
489 tGATT_STATUS att_ret = GATT_SUCCESS;
490
491 if (p_tcb != NULL) {
492 cmd_code &= ~GATT_AUTH_SIGN_MASK;
493
494 /* no pending request or value confirmation */
495 if (p_tcb->pending_cl_req == p_tcb->next_slot_inq ||
496 cmd_code == GATT_HANDLE_VALUE_CONF) {
497 att_ret = attp_send_msg_to_l2cap(p_tcb, p_cmd);
498 if (att_ret == GATT_CONGESTED || att_ret == GATT_SUCCESS) {
499 /* do not enq cmd if handle value confirmation or set request */
500 if (cmd_code != GATT_HANDLE_VALUE_CONF && cmd_code != GATT_CMD_WRITE) {
501 gatt_start_rsp_timer (clcb_idx);
502 gatt_cmd_enq(p_tcb, clcb_idx, 0, cmd_code, NULL);
503 }
504 } else {
505 att_ret = GATT_INTERNAL_ERROR;
506 }
507 } else {
508 att_ret = GATT_CMD_STARTED;
509 gatt_cmd_enq(p_tcb, clcb_idx, 1, cmd_code, p_cmd);
510 }
511 } else {
512 att_ret = GATT_ERROR;
513 }
514
515 return att_ret;
516 }
517 /*******************************************************************************
518 **
519 ** Function attp_send_cl_msg
520 **
521 ** Description This function sends the client request or confirmation message
522 ** to server.
523 **
524 ** Parameter p_tcb: pointer to the connection control block.
525 ** clcb_idx: clcb index
526 ** op_code: message op code.
527 ** p_msg: pointer to message parameters structure.
528 **
529 ** Returns GATT_SUCCESS if successfully sent; otherwise error code.
530 **
531 **
532 *******************************************************************************/
attp_send_cl_msg(tGATT_TCB * p_tcb,UINT16 clcb_idx,UINT8 op_code,tGATT_CL_MSG * p_msg)533 tGATT_STATUS attp_send_cl_msg (tGATT_TCB *p_tcb, UINT16 clcb_idx, UINT8 op_code, tGATT_CL_MSG *p_msg)
534 {
535 tGATT_STATUS status = GATT_NO_RESOURCES;
536 BT_HDR *p_cmd = NULL;
537 UINT16 offset = 0, handle;
538
539 if (p_tcb != NULL) {
540 switch (op_code) {
541 case GATT_REQ_MTU:
542 if (p_msg->mtu <= GATT_MAX_MTU_SIZE) {
543 p_tcb->payload_size = p_msg->mtu;
544 p_cmd = attp_build_mtu_cmd(GATT_REQ_MTU, p_msg->mtu);
545 } else {
546 status = GATT_ILLEGAL_PARAMETER;
547 }
548 break;
549
550 case GATT_REQ_FIND_INFO:
551 case GATT_REQ_READ_BY_TYPE:
552 case GATT_REQ_READ_BY_GRP_TYPE:
553 if (GATT_HANDLE_IS_VALID (p_msg->browse.s_handle) &&
554 GATT_HANDLE_IS_VALID (p_msg->browse.e_handle) &&
555 p_msg->browse.s_handle <= p_msg->browse.e_handle) {
556 p_cmd = attp_build_browse_cmd(op_code,
557 p_msg->browse.s_handle,
558 p_msg->browse.e_handle,
559 p_msg->browse.uuid);
560 } else {
561 status = GATT_ILLEGAL_PARAMETER;
562 }
563 break;
564
565 case GATT_REQ_READ_BLOB:
566 offset = p_msg->read_blob.offset;
567 /* fall through */
568 case GATT_REQ_READ:
569 handle = (op_code == GATT_REQ_READ) ? p_msg->handle : p_msg->read_blob.handle;
570 /* handle checking */
571 if (GATT_HANDLE_IS_VALID (handle)) {
572 p_cmd = attp_build_handle_cmd(op_code, handle, offset);
573 } else {
574 status = GATT_ILLEGAL_PARAMETER;
575 }
576 break;
577
578 case GATT_HANDLE_VALUE_CONF:
579 p_cmd = attp_build_opcode_cmd(op_code);
580 break;
581
582 case GATT_REQ_PREPARE_WRITE:
583 offset = p_msg->attr_value.offset;
584 /* fall through */
585 case GATT_REQ_WRITE:
586 case GATT_CMD_WRITE:
587 case GATT_SIGN_CMD_WRITE:
588 if (GATT_HANDLE_IS_VALID (p_msg->attr_value.handle)) {
589 p_cmd = attp_build_value_cmd (p_tcb->payload_size,
590 op_code, p_msg->attr_value.handle,
591 offset,
592 p_msg->attr_value.len,
593 p_msg->attr_value.value);
594 } else {
595 status = GATT_ILLEGAL_PARAMETER;
596 }
597 break;
598
599 case GATT_REQ_EXEC_WRITE:
600 p_cmd = attp_build_exec_write_cmd(op_code, p_msg->exec_write);
601 break;
602
603 case GATT_REQ_FIND_TYPE_VALUE:
604 p_cmd = attp_build_read_by_type_value_cmd(p_tcb->payload_size, &p_msg->find_type_value);
605 break;
606
607 case GATT_REQ_READ_MULTI:
608 p_cmd = attp_build_read_multi_cmd(p_tcb->payload_size,
609 p_msg->read_multi.num_handles,
610 p_msg->read_multi.handles);
611 break;
612
613 default:
614 break;
615 }
616
617 if (p_cmd != NULL) {
618 status = attp_cl_send_cmd(p_tcb, clcb_idx, op_code, p_cmd);
619 }
620
621 } else {
622 GATT_TRACE_ERROR("Peer device not connected");
623 }
624
625 return status;
626 }
627 #endif
628