| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329 |
- /******************************************************************************
- *
- * Copyright (C) 2006-2013 Broadcom Corporation
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at:
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- *
- ******************************************************************************/
- #include <string.h>
- #include "common/bt_target.h"
- #include "stack/avrc_api.h"
- #include "stack/avrc_defs.h"
- #include "avrc_int.h"
- #include "osi/allocator.h"
- #if (defined(AVRC_INCLUDED) && AVRC_INCLUDED == TRUE)
- /*****************************************************************************
- ** Global data
- *****************************************************************************/
- #if (AVRC_METADATA_INCLUDED == TRUE)
- /*******************************************************************************
- **
- ** Function avrc_bld_next_cmd
- **
- ** Description This function builds the Request Continue or Abort command.
- **
- ** Returns AVRC_STS_NO_ERROR, if the command is built successfully
- ** Otherwise, the error code.
- **
- *******************************************************************************/
- static tAVRC_STS avrc_bld_next_cmd (tAVRC_NEXT_CMD *p_cmd, BT_HDR *p_pkt)
- {
- UINT8 *p_data, *p_start;
- AVRC_TRACE_API("avrc_bld_next_cmd");
- /* get the existing length, if any, and also the num attributes */
- p_start = (UINT8 *)(p_pkt + 1) + p_pkt->offset;
- p_data = p_start + 2; /* pdu + rsvd */
- /* add fixed lenth 1 - pdu_id (1) */
- UINT16_TO_BE_STREAM(p_data, 1);
- UINT8_TO_BE_STREAM(p_data, p_cmd->target_pdu);
- p_pkt->len = (p_data - p_start);
- return AVRC_STS_NO_ERROR;
- }
- /*****************************************************************************
- ** the following commands are introduced in AVRCP 1.4
- *****************************************************************************/
- #if (AVRC_ADV_CTRL_INCLUDED == TRUE)
- /*******************************************************************************
- **
- ** Function avrc_bld_set_abs_volume_cmd
- **
- ** Description This function builds the Set Absolute Volume command.
- **
- ** Returns AVRC_STS_NO_ERROR, if the command is built successfully
- ** Otherwise, the error code.
- **
- *******************************************************************************/
- static tAVRC_STS avrc_bld_set_abs_volume_cmd (tAVRC_SET_VOLUME_CMD *p_cmd, BT_HDR *p_pkt)
- {
- UINT8 *p_data, *p_start;
- AVRC_TRACE_API("avrc_bld_set_abs_volume_cmd");
- /* get the existing length, if any, and also the num attributes */
- p_start = (UINT8 *)(p_pkt + 1) + p_pkt->offset;
- p_data = p_start + 2; /* pdu + rsvd */
- /* add fixed lenth 1 - volume (1) */
- UINT16_TO_BE_STREAM(p_data, 1);
- UINT8_TO_BE_STREAM(p_data, (AVRC_MAX_VOLUME & p_cmd->volume));
- p_pkt->len = (p_data - p_start);
- return AVRC_STS_NO_ERROR;
- }
- /*******************************************************************************
- **
- ** Function avrc_bld_vol_change_notfn
- **
- ** Description This function builds the register notification for volume change.
- **
- ** Returns AVRC_STS_NO_ERROR, if the command is built successfully
- ** Otherwise, the error code.
- **
- *******************************************************************************/
- static tAVRC_STS avrc_bld_register_change_notfn(UINT8 event_id, UINT32 event_parameter, BT_HDR *p_pkt)
- {
- UINT8 *p_data, *p_start;
- AVRC_TRACE_API("avrc_bld_vol_change");
- /* get the existing length, if any, and also the num attributes */
- // Set the notify value
- p_start = (UINT8 *)(p_pkt + 1) + p_pkt->offset;
- p_data = p_start + 2; /* pdu + rsvd */
- /* add fixed length 5 -*/
- UINT16_TO_BE_STREAM(p_data, 5);
- UINT8_TO_BE_STREAM(p_data, event_id);
- UINT32_TO_BE_STREAM(p_data, event_parameter);
- p_pkt->len = (p_data - p_start);
- return AVRC_STS_NO_ERROR;
- }
- #endif
- /*******************************************************************************
- **
- ** Function avrc_bld_init_cmd_buffer
- **
- ** Description This function initializes the command buffer based on PDU
- **
- ** Returns NULL, if no GKI buffer or failure to build the message.
- ** Otherwise, the GKI buffer that contains the initialized message.
- **
- *******************************************************************************/
- static BT_HDR *avrc_bld_init_cmd_buffer(tAVRC_COMMAND *p_cmd)
- {
- UINT8 opcode = avrc_opcode_from_pdu(p_cmd->pdu);
- AVRC_TRACE_API("avrc_bld_init_cmd_buffer: pdu=%x, opcode=%x", p_cmd->pdu, opcode);
- UINT16 offset = 0;
- switch (opcode) {
- case AVRC_OP_PASS_THRU:
- offset = AVRC_MSG_PASS_THRU_OFFSET;
- break;
- case AVRC_OP_VENDOR:
- offset = AVRC_MSG_VENDOR_OFFSET;
- break;
- }
- /* allocate and initialize the buffer */
- BT_HDR *p_pkt = (BT_HDR *)osi_malloc(AVRC_META_CMD_BUF_SIZE);
- if (p_pkt) {
- UINT8 *p_data, *p_start;
- p_pkt->layer_specific = AVCT_DATA_CTRL;
- p_pkt->event = opcode;
- p_pkt->offset = offset;
- p_data = (UINT8 *)(p_pkt + 1) + p_pkt->offset;
- p_start = p_data;
- /* pass thru - group navigation - has a two byte op_id, so dont do it here */
- if (opcode != AVRC_OP_PASS_THRU) {
- *p_data++ = p_cmd->pdu;
- }
- switch (opcode) {
- case AVRC_OP_VENDOR:
- /* reserved 0, packet_type 0 */
- UINT8_TO_BE_STREAM(p_data, 0);
- /* continue to the next "case to add length */
- /* add fixed lenth - 0 */
- UINT16_TO_BE_STREAM(p_data, 0);
- break;
- }
- p_pkt->len = (p_data - p_start);
- }
- p_cmd->cmd.opcode = opcode;
- return p_pkt;
- }
- /*******************************************************************************
- **
- ** Function avrc_bld_set_player_value_cmd
- **
- ** Description This function builds the Set Player Application Value command.
- **
- ** Returns AVRC_STS_NO_ERROR, if the command is built successfully
- ** Otherwise, the error code.
- **
- *******************************************************************************/
- static tAVRC_STS avrc_bld_set_player_value_cmd(tAVRC_SET_APP_VALUE_CMD *p_cmd, BT_HDR *p_pkt)
- {
- UINT8 *p_data, *p_start;
- /* get the existing length, if any, and also the num attributes */
- p_start = (UINT8 *)(p_pkt + 1) + p_pkt->offset;
- p_data = p_start + 2; /* pdu + rsvd */
- /* add length */
- UINT16_TO_BE_STREAM(p_data, 3);
- /* Number of attributes */
- UINT8_TO_BE_STREAM(p_data, 1);
- UINT8_TO_BE_STREAM(p_data, p_cmd->p_vals->attr_id);
- UINT8_TO_BE_STREAM(p_data, p_cmd->p_vals->attr_val);
- p_pkt->len = (p_data - p_start);
- return AVRC_STS_NO_ERROR;
- }
- /*******************************************************************************
- **
- ** Function avrc_bld_get_element_attr_cmd
- **
- ** Description This function builds the Get Element Attribute command.
- **
- ** Returns AVRC_STS_NO_ERROR, if the command is built successfully
- ** Otherwise, the error code.
- **
- *******************************************************************************/
- static tAVRC_STS avrc_bld_get_element_attr_cmd (tAVRC_GET_ELEM_ATTRS_CMD *p_cmd, BT_HDR *p_pkt)
- {
- int i;
- UINT8 *p_data, *p_start;
- AVRC_TRACE_API("avrc_bld_get_element_attr_cmd num_attr: %d", p_cmd->num_attr);
- /* get the existing length, if any, and also the num attributes */
- p_start = (UINT8 *)(p_pkt + 1) + p_pkt->offset;
- p_data = p_start + 2; /* pdu + rsvd */
- /* add length */
- UINT16_TO_BE_STREAM(p_data, 8 + 1 /* id + attr count */ + p_cmd->num_attr * sizeof(UINT32));
- /* Identifier 0x0 (PLAYING) */
- UINT64_TO_BE_STREAM(p_data, (UINT64)(0));
- /* Attribute count */
- UINT8_TO_BE_STREAM(p_data, p_cmd->num_attr);
- for (i = 0; i < p_cmd->num_attr; i++) {
- AVRC_TRACE_API("avrc_bld_get_element_attr_cmd attr_id: %d", p_cmd->attrs[i]);
- UINT32_TO_BE_STREAM(p_data, p_cmd->attrs[i]);
- }
- p_pkt->len = (p_data - p_start);
- return AVRC_STS_NO_ERROR;
- }
- static tAVRC_STS avrc_bld_get_caps_cmd(tAVRC_GET_CAPS_CMD *p_cmd, BT_HDR *p_pkt)
- {
- UINT8 *p_data, *p_start;
- AVRC_TRACE_API("avrc_bld_get_caps");
- /* get the existing length, if any, and also the num attributes */
- // Set the notify value
- p_start = (UINT8 *)(p_pkt + 1) + p_pkt->offset;
- p_data = p_start + 2; /* pdu + rsvd */
- /* add fixed length 1 */
- UINT16_TO_BE_STREAM(p_data, 1);
- /* capability id */
- UINT8_TO_BE_STREAM(p_data, p_cmd->capability_id);
- p_pkt->len = (p_data - p_start);
- return AVRC_STS_NO_ERROR;
- }
- /*******************************************************************************
- **
- ** Function AVRC_BldCommand
- **
- ** Description This function builds the given AVRCP command to the given
- ** GKI buffer
- **
- ** Returns AVRC_STS_NO_ERROR, if the command is built successfully
- ** Otherwise, the error code.
- **
- *******************************************************************************/
- tAVRC_STS AVRC_BldCommand( tAVRC_COMMAND *p_cmd, BT_HDR **pp_pkt)
- {
- tAVRC_STS status = AVRC_STS_BAD_PARAM;
- BT_HDR *p_pkt;
- BOOLEAN alloc = FALSE;
- AVRC_TRACE_API("AVRC_BldCommand: pdu=%x status=%x", p_cmd->cmd.pdu, p_cmd->cmd.status);
- if (!p_cmd || !pp_pkt) {
- AVRC_TRACE_API("AVRC_BldCommand. Invalid parameters passed. p_cmd=%p, pp_pkt=%p",
- p_cmd, pp_pkt);
- return AVRC_STS_BAD_PARAM;
- }
- if (*pp_pkt == NULL) {
- if ((*pp_pkt = avrc_bld_init_cmd_buffer(p_cmd)) == NULL) {
- AVRC_TRACE_API("AVRC_BldCommand: Failed to initialize command buffer");
- return AVRC_STS_INTERNAL_ERR;
- }
- alloc = TRUE;
- }
- status = AVRC_STS_NO_ERROR;
- p_pkt = *pp_pkt;
- switch (p_cmd->pdu) {
- case AVRC_PDU_REQUEST_CONTINUATION_RSP: /* 0x40 */
- status = avrc_bld_next_cmd(&p_cmd->continu, p_pkt);
- break;
- case AVRC_PDU_ABORT_CONTINUATION_RSP: /* 0x41 */
- status = avrc_bld_next_cmd(&p_cmd->abort, p_pkt);
- break;
- #if (AVRC_ADV_CTRL_INCLUDED == TRUE)
- case AVRC_PDU_SET_ABSOLUTE_VOLUME: /* 0x50 */
- status = avrc_bld_set_abs_volume_cmd(&p_cmd->volume, p_pkt);
- break;
- #endif
- case AVRC_PDU_SET_PLAYER_APP_VALUE: /* 0x14 */
- status = avrc_bld_set_player_value_cmd(&p_cmd->set_app_val, p_pkt);
- break;
- case AVRC_PDU_GET_ELEMENT_ATTR: /* 0x20 */
- status = avrc_bld_get_element_attr_cmd(&p_cmd->get_elem_attrs, p_pkt);
- break;
- case AVRC_PDU_REGISTER_NOTIFICATION: /* 0x31 */
- status = avrc_bld_register_change_notfn(p_cmd->reg_notif.event_id, p_cmd->reg_notif.param, p_pkt);
- break;
- case AVRC_PDU_GET_CAPABILITIES:
- status = avrc_bld_get_caps_cmd(&p_cmd->get_caps, p_pkt);
- break;
- }
- if (alloc && (status != AVRC_STS_NO_ERROR) ) {
- osi_free(p_pkt);
- *pp_pkt = NULL;
- }
- AVRC_TRACE_API("AVRC_BldCommand: returning %d", status);
- return status;
- }
- #endif /* (AVRC_METADATA_INCLUDED == TRUE) */
- #endif /* #if (defined(AVRC_INCLUDED) && AVRC_INCLUDED == TRUE) */
|