| /****************************************************************************** |
| * |
| * 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 functions that handle the database |
| * |
| ******************************************************************************/ |
| |
| #include <stdlib.h> |
| #include <string.h> |
| #include <stdio.h> |
| |
| #include "bt_target.h" |
| |
| #include "gki.h" |
| |
| #include "l2cdefs.h" |
| #include "hcidefs.h" |
| #include "hcimsgs.h" |
| |
| #include "sdp_api.h" |
| #include "sdpint.h" |
| #include "wbt_api.h" |
| |
| #if SDP_SERVER_ENABLED == TRUE |
| /********************************************************************************/ |
| /* L O C A L F U N C T I O N P R O T O T Y P E S */ |
| /********************************************************************************/ |
| static BOOLEAN find_uuid_in_seq (UINT8 *p , UINT32 seq_len, UINT8 *p_his_uuid, |
| UINT16 his_len, int nest_level); |
| |
| |
| /******************************************************************************* |
| ** |
| ** Function sdp_db_service_search |
| ** |
| ** Description This function searches for a record that contains the |
| ** specified UIDs. It is passed either NULL to start at the |
| ** beginning, or the previous record found. |
| ** |
| ** Returns Pointer to the record, or NULL if not found. |
| ** |
| *******************************************************************************/ |
| tSDP_RECORD *sdp_db_service_search (tSDP_RECORD *p_rec, tSDP_UUID_SEQ *p_seq) |
| { |
| UINT16 xx, yy; |
| tSDP_ATTRIBUTE *p_attr; |
| tSDP_RECORD *p_end = &sdp_cb.server_db.record[sdp_cb.server_db.num_records]; |
| |
| /* If NULL, start at the beginning, else start at the first specified record */ |
| if (!p_rec) |
| p_rec = &sdp_cb.server_db.record[0]; |
| else |
| p_rec++; |
| |
| /* Look through the records. The spec says that a match occurs if */ |
| /* the record contains all the passed UUIDs in it. */ |
| for ( ; p_rec < p_end; p_rec++) |
| { |
| for (yy = 0; yy < p_seq->num_uids; yy++) |
| { |
| p_attr = &p_rec->attribute[0]; |
| for (xx = 0; xx < p_rec->num_attributes; xx++, p_attr++) |
| { |
| if (p_attr->type == UUID_DESC_TYPE) |
| { |
| if (sdpu_compare_uuid_arrays (p_attr->value_ptr, p_attr->len, |
| &p_seq->uuid_entry[yy].value[0], |
| p_seq->uuid_entry[yy].len)) |
| break; |
| } |
| else if (p_attr->type == DATA_ELE_SEQ_DESC_TYPE) |
| { |
| if (find_uuid_in_seq (p_attr->value_ptr, p_attr->len, |
| &p_seq->uuid_entry[yy].value[0], |
| p_seq->uuid_entry[yy].len, 0)) |
| break; |
| } |
| } |
| /* If any UUID was not found, on to the next record */ |
| if (xx == p_rec->num_attributes) |
| break; |
| } |
| |
| /* If every UUID was found in the record, return the record */ |
| if (yy == p_seq->num_uids) |
| return (p_rec); |
| } |
| |
| /* If here, no more records found */ |
| return (NULL); |
| } |
| |
| /******************************************************************************* |
| ** |
| ** Function find_uuid_in_seq |
| ** |
| ** Description This function searches a data element sequenct for a UUID. |
| ** |
| ** Returns TRUE if found, else FALSE |
| ** |
| *******************************************************************************/ |
| static BOOLEAN find_uuid_in_seq (UINT8 *p , UINT32 seq_len, UINT8 *p_uuid, |
| UINT16 uuid_len, int nest_level) |
| { |
| UINT8 *p_end = p + seq_len; |
| UINT8 type; |
| UINT32 len; |
| |
| /* A little safety check to avoid excessive recursion */ |
| if (nest_level > 3) |
| return (FALSE); |
| |
| while (p < p_end) |
| { |
| type = *p++; |
| p = sdpu_get_len_from_type (p, type, &len); |
| type = type >> 3; |
| if (type == UUID_DESC_TYPE) |
| { |
| if (sdpu_compare_uuid_arrays (p, len, p_uuid, uuid_len)) |
| return (TRUE); |
| } |
| else if (type == DATA_ELE_SEQ_DESC_TYPE) |
| { |
| if (find_uuid_in_seq (p, len, p_uuid, uuid_len, nest_level + 1)) |
| return (TRUE); |
| } |
| p = p + len; |
| } |
| |
| /* If here, failed to match */ |
| return (FALSE); |
| } |
| |
| /******************************************************************************* |
| ** |
| ** Function sdp_db_find_record |
| ** |
| ** Description This function searches for a record with a specific handle |
| ** It is passed the handle of the record. |
| ** |
| ** Returns Pointer to the record, or NULL if not found. |
| ** |
| *******************************************************************************/ |
| tSDP_RECORD *sdp_db_find_record (UINT32 handle) |
| { |
| tSDP_RECORD *p_rec; |
| tSDP_RECORD *p_end = &sdp_cb.server_db.record[sdp_cb.server_db.num_records]; |
| |
| /* Look through the records for the caller's handle */ |
| for (p_rec = &sdp_cb.server_db.record[0]; p_rec < p_end; p_rec++) |
| { |
| if (p_rec->record_handle == handle) |
| return (p_rec); |
| } |
| |
| /* Record with that handle not found. */ |
| return (NULL); |
| } |
| |
| /******************************************************************************* |
| ** |
| ** Function sdp_db_find_attr_in_rec |
| ** |
| ** Description This function searches a record for specific attributes. |
| ** It is passed a pointer to the record. If the record contains |
| ** the specified attribute, (the caller may specify be a range |
| ** of attributes), the attribute is returned. |
| ** |
| ** Returns Pointer to the attribute, or NULL if not found. |
| ** |
| *******************************************************************************/ |
| tSDP_ATTRIBUTE *sdp_db_find_attr_in_rec (tSDP_RECORD *p_rec, UINT16 start_attr, |
| UINT16 end_attr) |
| { |
| tSDP_ATTRIBUTE *p_at; |
| UINT16 xx; |
| |
| /* Note that the attributes in a record are assumed to be in sorted order */ |
| for (xx = 0, p_at = &p_rec->attribute[0]; xx < p_rec->num_attributes; |
| xx++, p_at++) |
| { |
| if ((p_at->id >= start_attr) && (p_at->id <= end_attr)) |
| return (p_at); |
| } |
| |
| /* No matching attribute found */ |
| return (NULL); |
| } |
| |
| |
| /******************************************************************************* |
| ** |
| ** Function sdp_compose_proto_list |
| ** |
| ** Description This function is called to compose a data sequence from |
| ** protocol element list struct pointer |
| ** |
| ** Returns the length of the data sequence |
| ** |
| *******************************************************************************/ |
| static int sdp_compose_proto_list( UINT8 *p, UINT16 num_elem, |
| tSDP_PROTOCOL_ELEM *p_elem_list) |
| { |
| UINT16 xx, yy, len; |
| BOOLEAN is_rfcomm_scn; |
| UINT8 *p_head = p; |
| UINT8 *p_len; |
| |
| /* First, build the protocol list. This consists of a set of data element |
| ** sequences, one for each layer. Each layer sequence consists of layer's |
| ** UUID and optional parameters |
| */ |
| for (xx = 0; xx < num_elem; xx++, p_elem_list++) |
| { |
| len = 3 + (p_elem_list->num_params * 3); |
| UINT8_TO_BE_STREAM (p, (DATA_ELE_SEQ_DESC_TYPE << 3) | SIZE_IN_NEXT_BYTE); |
| |
| p_len = p; |
| *p++ = (UINT8) len; |
| |
| UINT8_TO_BE_STREAM (p, (UUID_DESC_TYPE << 3) | SIZE_TWO_BYTES); |
| UINT16_TO_BE_STREAM (p, p_elem_list->protocol_uuid); |
| |
| if (p_elem_list->protocol_uuid == UUID_PROTOCOL_RFCOMM) |
| is_rfcomm_scn = TRUE; |
| else |
| is_rfcomm_scn = FALSE; |
| |
| for (yy = 0; yy < p_elem_list->num_params; yy++) |
| { |
| if (is_rfcomm_scn) |
| { |
| UINT8_TO_BE_STREAM (p, (UINT_DESC_TYPE << 3) | SIZE_ONE_BYTE); |
| UINT8_TO_BE_STREAM (p, p_elem_list->params[yy]); |
| |
| *p_len -= 1; |
| } |
| else |
| { |
| UINT8_TO_BE_STREAM (p, (UINT_DESC_TYPE << 3) | SIZE_TWO_BYTES); |
| UINT16_TO_BE_STREAM (p, p_elem_list->params[yy]); |
| } |
| } |
| } |
| return (p - p_head); |
| } |
| |
| #endif /* SDP_SERVER_ENABLED == TRUE */ |
| |
| /******************************************************************************* |
| ** |
| ** Function SDP_CreateRecord |
| ** |
| ** Description This function is called to create a record in the database. |
| ** This would be through the SDP database maintenance API. The |
| ** record is created empty, teh application should then call |
| ** "add_attribute" to add the record's attributes. |
| ** |
| ** Returns Record handle if OK, else 0. |
| ** |
| *******************************************************************************/ |
| UINT32 SDP_CreateRecord (void) |
| { |
| #if SDP_SERVER_ENABLED == TRUE |
| UINT32 handle; |
| UINT8 buf[4]; |
| tSDP_DB *p_db = &sdp_cb.server_db; |
| |
| /* First, check if there is a free record */ |
| if (p_db->num_records < SDP_MAX_RECORDS) |
| { |
| memset (&p_db->record[p_db->num_records], 0, |
| sizeof (tSDP_RECORD)); |
| |
| /* We will use a handle of the first unreserved handle plus last record |
| ** number + 1 */ |
| if (p_db->num_records) |
| handle = p_db->record[p_db->num_records - 1].record_handle + 1; |
| else |
| handle = 0x10000; |
| |
| p_db->record[p_db->num_records].record_handle = handle; |
| |
| p_db->num_records++; |
| SDP_TRACE_DEBUG1("SDP_CreateRecord ok, num_records:%d", p_db->num_records); |
| /* Add the first attribute (the handle) automatically */ |
| UINT32_TO_BE_FIELD (buf, handle); |
| SDP_AddAttribute (handle, ATTR_ID_SERVICE_RECORD_HDL, UINT_DESC_TYPE, |
| 4, buf); |
| |
| return (p_db->record[p_db->num_records - 1].record_handle); |
| } |
| else SDP_TRACE_ERROR1("SDP_CreateRecord fail, exceed maximum records:%d", SDP_MAX_RECORDS); |
| #endif |
| return (0); |
| } |
| |
| |
| /******************************************************************************* |
| ** |
| ** Function SDP_DeleteRecord |
| ** |
| ** Description This function is called to add a record (or all records) |
| ** from the database. This would be through the SDP database |
| ** maintenance API. |
| ** |
| ** If a record handle of 0 is passed, all records are deleted. |
| ** |
| ** Returns TRUE if succeeded, else FALSE |
| ** |
| *******************************************************************************/ |
| BOOLEAN SDP_DeleteRecord (UINT32 handle) |
| { |
| #if SDP_SERVER_ENABLED == TRUE |
| UINT16 xx, yy, zz; |
| tSDP_RECORD *p_rec = &sdp_cb.server_db.record[0]; |
| |
| if (handle == 0 || sdp_cb.server_db.num_records == 0) |
| { |
| /* Delete all records in the database */ |
| sdp_cb.server_db.num_records = 0; |
| |
| /* require new DI record to be created in SDP_SetLocalDiRecord */ |
| sdp_cb.server_db.di_primary_handle = 0; |
| sdp_cb.server_db.brcm_di_registered = 0; |
| |
| return (TRUE); |
| } |
| else |
| { |
| /* Find the record in the database */ |
| for (xx = 0; xx < sdp_cb.server_db.num_records; xx++, p_rec++) |
| { |
| if (p_rec->record_handle == handle) |
| { |
| /* Found it. Shift everything up one */ |
| for (yy = xx; yy < sdp_cb.server_db.num_records; yy++, p_rec++) |
| { |
| *p_rec = *(p_rec + 1); |
| |
| /* Adjust the attribute value pointer for each attribute */ |
| for (zz = 0; zz < p_rec->num_attributes; zz++) |
| p_rec->attribute[zz].value_ptr -= sizeof(tSDP_RECORD); |
| } |
| |
| sdp_cb.server_db.num_records--; |
| |
| SDP_TRACE_DEBUG1("SDP_DeleteRecord ok, num_records:%d", sdp_cb.server_db.num_records); |
| /* if we're deleting the primary DI record, clear the */ |
| /* value in the control block */ |
| if( sdp_cb.server_db.di_primary_handle == handle ) |
| { |
| sdp_cb.server_db.di_primary_handle = 0; |
| sdp_cb.server_db.brcm_di_registered = 0; |
| } |
| |
| return (TRUE); |
| } |
| } |
| } |
| #endif |
| return (FALSE); |
| } |
| |
| |
| /******************************************************************************* |
| ** |
| ** Function SDP_AddAttribute |
| ** |
| ** Description This function is called to add an attribute to a record. |
| ** This would be through the SDP database maintenance API. |
| ** If the attribute already exists in the record, it is replaced |
| ** with the new value. |
| ** |
| ** NOTE Attribute values must be passed as a Big Endian stream. |
| ** |
| ** Returns TRUE if added OK, else FALSE |
| ** |
| *******************************************************************************/ |
| BOOLEAN SDP_AddAttribute (UINT32 handle, UINT16 attr_id, UINT8 attr_type, |
| UINT32 attr_len, UINT8 *p_val) |
| { |
| #if SDP_SERVER_ENABLED == TRUE |
| UINT16 xx, yy, zz; |
| tSDP_RECORD *p_rec = &sdp_cb.server_db.record[0]; |
| |
| #if (BT_TRACE_VERBOSE == TRUE) |
| if (sdp_cb.trace_level >= BT_TRACE_LEVEL_DEBUG) |
| { |
| if ((attr_type == UINT_DESC_TYPE) || |
| (attr_type == TWO_COMP_INT_DESC_TYPE) || |
| (attr_type == UUID_DESC_TYPE) || |
| (attr_type == DATA_ELE_SEQ_DESC_TYPE) || |
| (attr_type == DATA_ELE_ALT_DESC_TYPE)) |
| { |
| UINT8 num_array[400]; |
| UINT32 i; |
| UINT32 len = (attr_len > 200) ? 200 : attr_len; |
| |
| num_array[0] ='\0'; |
| for (i = 0; i < len; i++) |
| { |
| sprintf((char *)&num_array[i*2],"%02X",(UINT8)(p_val[i])); |
| } |
| SDP_TRACE_DEBUG6("SDP_AddAttribute: handle:%X, id:%04X, type:%d, len:%d, p_val:%p, *p_val:%s", |
| handle,attr_id,attr_type,attr_len,p_val,num_array); |
| } |
| else if (attr_type == BOOLEAN_DESC_TYPE) |
| { |
| SDP_TRACE_DEBUG6("SDP_AddAttribute: handle:%X, id:%04X, type:%d, len:%d, p_val:%p, *p_val:%d", |
| handle,attr_id,attr_type,attr_len,p_val,*p_val); |
| } |
| else |
| { |
| SDP_TRACE_DEBUG6("SDP_AddAttribute: handle:%X, id:%04X, type:%d, len:%d, p_val:%p, *p_val:%s", |
| handle,attr_id,attr_type,attr_len,p_val,p_val); |
| } |
| } |
| #endif |
| |
| /* Find the record in the database */ |
| for (zz = 0; zz < sdp_cb.server_db.num_records; zz++, p_rec++) |
| { |
| if (p_rec->record_handle == handle) |
| { |
| tSDP_ATTRIBUTE *p_attr = &p_rec->attribute[0]; |
| |
| /* Found the record. Now, see if the attribute already exists */ |
| for (xx = 0; xx < p_rec->num_attributes; xx++, p_attr++) |
| { |
| /* The attribute exists. replace it */ |
| if (p_attr->id == attr_id) |
| { |
| SDP_DeleteAttribute (handle, attr_id); |
| break; |
| } |
| if (p_attr->id > attr_id) |
| break; |
| } |
| |
| if (p_rec->num_attributes == SDP_MAX_REC_ATTR) |
| return (FALSE); |
| |
| /* If not found, see if we can allocate a new entry */ |
| if (xx == p_rec->num_attributes) |
| p_attr = &p_rec->attribute[p_rec->num_attributes]; |
| else |
| { |
| /* Since the attributes are kept in sorted order, insert ours here */ |
| for (yy = p_rec->num_attributes; yy > xx; yy--) |
| p_rec->attribute[yy] = p_rec->attribute[yy - 1]; |
| } |
| |
| p_attr->id = attr_id; |
| p_attr->type = attr_type; |
| p_attr->len = attr_len; |
| |
| if (p_rec->free_pad_ptr + attr_len >= SDP_MAX_PAD_LEN) |
| { |
| /* do truncate only for text string type descriptor */ |
| if (attr_type == TEXT_STR_DESC_TYPE) |
| { |
| SDP_TRACE_WARNING2("SDP_AddAttribute: attr_len:%d too long. truncate to (%d)", |
| attr_len, SDP_MAX_PAD_LEN - p_rec->free_pad_ptr ); |
| |
| attr_len = SDP_MAX_PAD_LEN - p_rec->free_pad_ptr; |
| p_val[SDP_MAX_PAD_LEN - p_rec->free_pad_ptr] = '\0'; |
| p_val[SDP_MAX_PAD_LEN - p_rec->free_pad_ptr+1] = '\0'; |
| } |
| else |
| attr_len = 0; |
| } |
| |
| if ((attr_len > 0) && (p_val != 0)) |
| { |
| p_attr->len = attr_len; |
| memcpy (&p_rec->attr_pad[p_rec->free_pad_ptr], p_val, (size_t)attr_len); |
| p_attr->value_ptr = &p_rec->attr_pad[p_rec->free_pad_ptr]; |
| p_rec->free_pad_ptr += attr_len; |
| } |
| else if ((attr_len == 0 && p_attr->len != 0) || /* if truncate to 0 length, simply don't add */ |
| p_val == 0) |
| { |
| SDP_TRACE_ERROR2("SDP_AddAttribute fail, length exceed maximum: ID %d: attr_len:%d ", |
| attr_id, attr_len ); |
| p_attr->id = p_attr->type = p_attr->len = 0; |
| return (FALSE); |
| } |
| p_rec->num_attributes++; |
| |
| /*** Mark DI record as used by Broadcom ***/ |
| if (handle == sdp_cb.server_db.di_primary_handle && |
| attr_id == ATTR_ID_EXT_BRCM_VERSION) |
| sdp_cb.server_db.brcm_di_registered = TRUE; |
| |
| return (TRUE); |
| } |
| } |
| #endif |
| return (FALSE); |
| } |
| |
| |
| /******************************************************************************* |
| ** |
| ** Function SDP_AddSequence |
| ** |
| ** Description This function is called to add a sequence to a record. |
| ** This would be through the SDP database maintenance API. |
| ** If the sequence already exists in the record, it is replaced |
| ** with the new sequence. |
| ** |
| ** NOTE Element values must be passed as a Big Endian stream. |
| ** |
| ** Returns TRUE if added OK, else FALSE |
| ** |
| *******************************************************************************/ |
| BOOLEAN SDP_AddSequence (UINT32 handle, UINT16 attr_id, UINT16 num_elem, |
| UINT8 type[], UINT8 len[], UINT8 *p_val[]) |
| { |
| #if SDP_SERVER_ENABLED == TRUE |
| UINT16 xx; |
| UINT8 *p_buff; |
| UINT8 *p; |
| UINT8 *p_head; |
| BOOLEAN result; |
| |
| if ((p_buff = (UINT8 *) GKI_getbuf(sizeof(UINT8) * SDP_MAX_ATTR_LEN * 2)) == NULL) |
| { |
| SDP_TRACE_ERROR0("SDP_AddSequence cannot get a buffer!"); |
| return (FALSE); |
| } |
| p = p_buff; |
| |
| /* First, build the sequence */ |
| for (xx = 0; xx < num_elem; xx++) |
| { |
| p_head = p; |
| switch (len[xx]) |
| { |
| case 1: |
| UINT8_TO_BE_STREAM (p, (type[xx] << 3) | SIZE_ONE_BYTE); |
| break; |
| case 2: |
| UINT8_TO_BE_STREAM (p, (type[xx] << 3) | SIZE_TWO_BYTES); |
| break; |
| case 4: |
| UINT8_TO_BE_STREAM (p, (type[xx] << 3) | SIZE_FOUR_BYTES); |
| break; |
| case 8: |
| UINT8_TO_BE_STREAM (p, (type[xx] << 3) | SIZE_EIGHT_BYTES); |
| break; |
| case 16: |
| UINT8_TO_BE_STREAM (p, (type[xx] << 3) | SIZE_SIXTEEN_BYTES); |
| break; |
| default: |
| UINT8_TO_BE_STREAM (p, (type[xx] << 3) | SIZE_IN_NEXT_BYTE); |
| UINT8_TO_BE_STREAM (p, len[xx]); |
| break; |
| } |
| |
| ARRAY_TO_BE_STREAM (p, p_val[xx], len[xx]); |
| |
| if (p - p_buff > SDP_MAX_ATTR_LEN) |
| { |
| /* go back to before we add this element */ |
| p = p_head; |
| if(p_head == p_buff) |
| { |
| /* the first element exceed the max length */ |
| SDP_TRACE_ERROR0 ("SDP_AddSequence - too long(attribute is not added)!!"); |
| GKI_freebuf(p_buff); |
| return FALSE; |
| } |
| else |
| SDP_TRACE_ERROR2 ("SDP_AddSequence - too long, add %d elements of %d", xx, num_elem); |
| break; |
| } |
| } |
| result = SDP_AddAttribute (handle, attr_id, DATA_ELE_SEQ_DESC_TYPE,(UINT32) (p - p_buff), p_buff); |
| GKI_freebuf(p_buff); |
| return result; |
| #else /* SDP_SERVER_ENABLED == FALSE */ |
| return (FALSE); |
| #endif |
| } |
| |
| |
| /******************************************************************************* |
| ** |
| ** Function SDP_AddUuidSequence |
| ** |
| ** Description This function is called to add a UUID sequence to a record. |
| ** This would be through the SDP database maintenance API. |
| ** If the sequence already exists in the record, it is replaced |
| ** with the new sequence. |
| ** |
| ** Returns TRUE if added OK, else FALSE |
| ** |
| *******************************************************************************/ |
| BOOLEAN SDP_AddUuidSequence (UINT32 handle, UINT16 attr_id, UINT16 num_uuids, |
| UINT16 *p_uuids) |
| { |
| #if SDP_SERVER_ENABLED == TRUE |
| UINT16 xx; |
| UINT8 *p_buff; |
| UINT8 *p; |
| INT32 max_len = SDP_MAX_ATTR_LEN -3; |
| BOOLEAN result; |
| |
| if ((p_buff = (UINT8 *) GKI_getbuf(sizeof(UINT8) * SDP_MAX_ATTR_LEN * 2)) == NULL) |
| { |
| SDP_TRACE_ERROR0("SDP_AddUuidSequence cannot get a buffer!"); |
| return (FALSE); |
| } |
| p = p_buff; |
| |
| /* First, build the sequence */ |
| for (xx = 0; xx < num_uuids ; xx++, p_uuids++) |
| { |
| UINT8_TO_BE_STREAM (p, (UUID_DESC_TYPE << 3) | SIZE_TWO_BYTES); |
| UINT16_TO_BE_STREAM (p, *p_uuids); |
| |
| if((p - p_buff) > max_len) |
| { |
| SDP_TRACE_WARNING2 ("SDP_AddUuidSequence - too long, add %d uuids of %d", xx, num_uuids); |
| break; |
| } |
| } |
| |
| result = SDP_AddAttribute (handle, attr_id, DATA_ELE_SEQ_DESC_TYPE,(UINT32) (p - p_buff), p_buff); |
| GKI_freebuf(p_buff); |
| return result; |
| #else /* SDP_SERVER_ENABLED == FALSE */ |
| return (FALSE); |
| #endif |
| } |
| |
| /******************************************************************************* |
| ** |
| ** Function SDP_AddProtocolList |
| ** |
| ** Description This function is called to add a protocol descriptor list to |
| ** a record. This would be through the SDP database maintenance API. |
| ** If the protocol list already exists in the record, it is replaced |
| ** with the new list. |
| ** |
| ** Returns TRUE if added OK, else FALSE |
| ** |
| *******************************************************************************/ |
| BOOLEAN SDP_AddProtocolList (UINT32 handle, UINT16 num_elem, |
| tSDP_PROTOCOL_ELEM *p_elem_list) |
| { |
| #if SDP_SERVER_ENABLED == TRUE |
| UINT8 *p_buff; |
| int offset; |
| BOOLEAN result; |
| |
| if ((p_buff = (UINT8 *) GKI_getbuf(sizeof(UINT8) * SDP_MAX_ATTR_LEN * 2)) == NULL) |
| { |
| SDP_TRACE_ERROR0("SDP_AddProtocolList cannot get a buffer!"); |
| return (FALSE); |
| } |
| |
| offset = sdp_compose_proto_list(p_buff, num_elem, p_elem_list); |
| result = SDP_AddAttribute (handle, ATTR_ID_PROTOCOL_DESC_LIST,DATA_ELE_SEQ_DESC_TYPE, (UINT32) offset, p_buff); |
| GKI_freebuf(p_buff); |
| return result; |
| #else /* SDP_SERVER_ENABLED == FALSE */ |
| return (FALSE); |
| #endif |
| } |
| |
| |
| /******************************************************************************* |
| ** |
| ** Function SDP_AddAdditionProtoLists |
| ** |
| ** Description This function is called to add a protocol descriptor list to |
| ** a record. This would be through the SDP database maintenance API. |
| ** If the protocol list already exists in the record, it is replaced |
| ** with the new list. |
| ** |
| ** Returns TRUE if added OK, else FALSE |
| ** |
| *******************************************************************************/ |
| BOOLEAN SDP_AddAdditionProtoLists (UINT32 handle, UINT16 num_elem, |
| tSDP_PROTO_LIST_ELEM *p_proto_list) |
| { |
| #if SDP_SERVER_ENABLED == TRUE |
| UINT16 xx; |
| UINT8 *p_buff; |
| UINT8 *p; |
| UINT8 *p_len; |
| int offset; |
| BOOLEAN result; |
| |
| if ((p_buff = (UINT8 *) GKI_getbuf(sizeof(UINT8) * SDP_MAX_ATTR_LEN * 2)) == NULL) |
| { |
| SDP_TRACE_ERROR0("SDP_AddAdditionProtoLists cannot get a buffer!"); |
| return (FALSE); |
| } |
| p = p_buff; |
| |
| /* for each ProtocolDescriptorList */ |
| for (xx = 0; xx < num_elem; xx++, p_proto_list++) |
| { |
| UINT8_TO_BE_STREAM (p, (DATA_ELE_SEQ_DESC_TYPE << 3) | SIZE_IN_NEXT_BYTE); |
| p_len = p++; |
| |
| offset = sdp_compose_proto_list(p, p_proto_list->num_elems, |
| p_proto_list->list_elem); |
| p += offset; |
| |
| *p_len = (UINT8)(p - p_len - 1); |
| } |
| result = SDP_AddAttribute (handle, ATTR_ID_ADDITION_PROTO_DESC_LISTS,DATA_ELE_SEQ_DESC_TYPE, |
| (UINT32) (p - p_buff), p_buff); |
| GKI_freebuf(p_buff); |
| return result; |
| |
| #else /* SDP_SERVER_ENABLED == FALSE */ |
| return (FALSE); |
| #endif |
| } |
| |
| /******************************************************************************* |
| ** |
| ** Function SDP_AddProfileDescriptorList |
| ** |
| ** Description This function is called to add a profile descriptor list to |
| ** a record. This would be through the SDP database maintenance API. |
| ** If the version already exists in the record, it is replaced |
| ** with the new one. |
| ** |
| ** Returns TRUE if added OK, else FALSE |
| ** |
| *******************************************************************************/ |
| BOOLEAN SDP_AddProfileDescriptorList (UINT32 handle, UINT16 profile_uuid, |
| UINT16 version) |
| { |
| #if SDP_SERVER_ENABLED == TRUE |
| UINT8 *p_buff; |
| UINT8 *p; |
| BOOLEAN result; |
| |
| if ((p_buff = (UINT8 *) GKI_getbuf(sizeof(UINT8) * SDP_MAX_ATTR_LEN)) == NULL) |
| { |
| SDP_TRACE_ERROR0("SDP_AddProfileDescriptorList cannot get a buffer!"); |
| return (FALSE); |
| } |
| p = p_buff+2; |
| |
| /* First, build the profile descriptor list. This consists of a data element sequence. */ |
| /* The sequence consists of profile's UUID and version number */ |
| UINT8_TO_BE_STREAM (p, (UUID_DESC_TYPE << 3) | SIZE_TWO_BYTES); |
| UINT16_TO_BE_STREAM (p, profile_uuid); |
| |
| UINT8_TO_BE_STREAM (p, (UINT_DESC_TYPE << 3) | SIZE_TWO_BYTES); |
| UINT16_TO_BE_STREAM (p, version); |
| |
| /* Add in type and length fields */ |
| *p_buff = (UINT8) ((DATA_ELE_SEQ_DESC_TYPE << 3) | SIZE_IN_NEXT_BYTE); |
| *(p_buff+1) = (UINT8) (p - (p_buff+2)); |
| |
| result = SDP_AddAttribute (handle, ATTR_ID_BT_PROFILE_DESC_LIST,DATA_ELE_SEQ_DESC_TYPE, (UINT32) (p - p_buff), p_buff); |
| GKI_freebuf(p_buff); |
| return result; |
| |
| #else /* SDP_SERVER_ENABLED == FALSE */ |
| return (FALSE); |
| #endif |
| } |
| |
| |
| /******************************************************************************* |
| ** |
| ** Function SDP_AddLanguageBaseAttrIDList |
| ** |
| ** Description This function is called to add a language base attr list to |
| ** a record. This would be through the SDP database maintenance API. |
| ** If the version already exists in the record, it is replaced |
| ** with the new one. |
| ** |
| ** Returns TRUE if added OK, else FALSE |
| ** |
| *******************************************************************************/ |
| BOOLEAN SDP_AddLanguageBaseAttrIDList (UINT32 handle, UINT16 lang, |
| UINT16 char_enc, UINT16 base_id) |
| { |
| #if SDP_SERVER_ENABLED == TRUE |
| UINT8 *p_buff; |
| UINT8 *p; |
| BOOLEAN result; |
| |
| if ((p_buff = (UINT8 *) GKI_getbuf(sizeof(UINT8) * SDP_MAX_ATTR_LEN)) == NULL) |
| { |
| SDP_TRACE_ERROR0("SDP_AddLanguageBaseAttrIDList cannot get a buffer!"); |
| return (FALSE); |
| } |
| p = p_buff; |
| |
| /* First, build the language base descriptor list. This consists of a data */ |
| /* element sequence. The sequence consists of 9 bytes (3 UINt16 fields) */ |
| UINT8_TO_BE_STREAM (p, (UINT_DESC_TYPE << 3) | SIZE_TWO_BYTES); |
| UINT16_TO_BE_STREAM (p, lang); |
| |
| UINT8_TO_BE_STREAM (p, (UINT_DESC_TYPE << 3) | SIZE_TWO_BYTES); |
| UINT16_TO_BE_STREAM (p, char_enc); |
| |
| UINT8_TO_BE_STREAM (p, (UINT_DESC_TYPE << 3) | SIZE_TWO_BYTES); |
| UINT16_TO_BE_STREAM (p, base_id); |
| |
| result = SDP_AddAttribute (handle, ATTR_ID_LANGUAGE_BASE_ATTR_ID_LIST,DATA_ELE_SEQ_DESC_TYPE, |
| (UINT32) (p - p_buff), p_buff); |
| GKI_freebuf(p_buff); |
| return result; |
| #else /* SDP_SERVER_ENABLED == FALSE */ |
| return (FALSE); |
| #endif |
| } |
| |
| |
| /******************************************************************************* |
| ** |
| ** Function SDP_AddServiceClassIdList |
| ** |
| ** Description This function is called to add a service list to a record. |
| ** This would be through the SDP database maintenance API. |
| ** If the service list already exists in the record, it is replaced |
| ** with the new list. |
| ** |
| ** Returns TRUE if added OK, else FALSE |
| ** |
| *******************************************************************************/ |
| BOOLEAN SDP_AddServiceClassIdList (UINT32 handle, UINT16 num_services, |
| UINT16 *p_service_uuids) |
| { |
| #if SDP_SERVER_ENABLED == TRUE |
| UINT16 xx; |
| UINT8 *p_buff; |
| UINT8 *p; |
| BOOLEAN result; |
| |
| if ((p_buff = (UINT8 *) GKI_getbuf(sizeof(UINT8) * SDP_MAX_ATTR_LEN * 2)) == NULL) |
| { |
| SDP_TRACE_ERROR0("SDP_AddServiceClassIdList cannot get a buffer!"); |
| return (FALSE); |
| } |
| p = p_buff; |
| |
| for (xx = 0; xx < num_services; xx++, p_service_uuids++) |
| { |
| UINT8_TO_BE_STREAM (p, (UUID_DESC_TYPE << 3) | SIZE_TWO_BYTES); |
| UINT16_TO_BE_STREAM (p, *p_service_uuids); |
| } |
| |
| result = SDP_AddAttribute (handle, ATTR_ID_SERVICE_CLASS_ID_LIST,DATA_ELE_SEQ_DESC_TYPE, |
| (UINT32) (p - p_buff), p_buff); |
| GKI_freebuf(p_buff); |
| return result; |
| #else /* SDP_SERVER_ENABLED == FALSE */ |
| return (FALSE); |
| #endif |
| } |
| |
| |
| /******************************************************************************* |
| ** |
| ** Function SDP_DeleteAttribute |
| ** |
| ** Description This function is called to delete an attribute from a record. |
| ** This would be through the SDP database maintenance API. |
| ** |
| ** Returns TRUE if deleted OK, else FALSE if not found |
| ** |
| *******************************************************************************/ |
| BOOLEAN SDP_DeleteAttribute (UINT32 handle, UINT16 attr_id) |
| { |
| #if SDP_SERVER_ENABLED == TRUE |
| UINT16 xx, yy; |
| tSDP_RECORD *p_rec = &sdp_cb.server_db.record[0]; |
| UINT8 *pad_ptr; |
| UINT32 len; /* Number of bytes in the entry */ |
| |
| /* Find the record in the database */ |
| for (xx = 0; xx < sdp_cb.server_db.num_records; xx++, p_rec++) |
| { |
| if (p_rec->record_handle == handle) |
| { |
| tSDP_ATTRIBUTE *p_attr = &p_rec->attribute[0]; |
| |
| SDP_TRACE_API2("Deleting attr_id 0x%04x for handle 0x%x", attr_id, handle); |
| /* Found it. Now, find the attribute */ |
| for (xx = 0; xx < p_rec->num_attributes; xx++, p_attr++) |
| { |
| if (p_attr->id == attr_id) |
| { |
| pad_ptr = p_attr->value_ptr; |
| len = p_attr->len; |
| |
| if (len) |
| { |
| for (yy = 0; yy < p_rec->num_attributes; yy++) |
| { |
| if( p_rec->attribute[yy].value_ptr > pad_ptr ) |
| p_rec->attribute[yy].value_ptr -= len; |
| } |
| } |
| |
| /* Found it. Shift everything up one */ |
| p_rec->num_attributes--; |
| |
| for (yy = xx; yy < p_rec->num_attributes; yy++, p_attr++) |
| { |
| *p_attr = *(p_attr + 1); |
| } |
| |
| /* adjust attribute values if needed */ |
| if (len) |
| { |
| xx = (p_rec->free_pad_ptr - ((pad_ptr+len) - |
| &p_rec->attr_pad[0])); |
| for( yy=0; yy<xx; yy++, pad_ptr++) |
| *pad_ptr = *(pad_ptr+len); |
| p_rec->free_pad_ptr -= len; |
| } |
| return (TRUE); |
| } |
| } |
| } |
| } |
| #endif |
| /* If here, not found */ |
| return (FALSE); |
| } |
| |
| /******************************************************************************* |
| ** |
| ** Function SDP_ReadRecord |
| ** |
| ** Description This function is called to get the raw data of the record |
| ** with the given handle from the database. |
| ** |
| ** Returns -1, if the record is not found. |
| ** Otherwise, the offset (0 or 1) to start of data in p_data. |
| ** |
| ** The size of data copied into p_data is in *p_data_len. |
| ** |
| *******************************************************************************/ |
| #if (SDP_RAW_DATA_INCLUDED == TRUE) |
| INT32 SDP_ReadRecord(UINT32 handle, UINT8 *p_data, INT32 *p_data_len) |
| { |
| INT32 len = 0; /* Number of bytes in the entry */ |
| INT32 offset = -1; /* default to not found */ |
| #if SDP_SERVER_ENABLED == TRUE |
| tSDP_RECORD *p_rec; |
| UINT16 start = 0; |
| UINT16 end = 0xffff; |
| tSDP_ATTRIBUTE *p_attr; |
| UINT16 rem_len; |
| UINT8 *p_rsp; |
| |
| /* Find the record in the database */ |
| p_rec = sdp_db_find_record(handle); |
| if(p_rec && p_data && p_data_len) |
| { |
| p_rsp = &p_data[3]; |
| while ( (p_attr = sdp_db_find_attr_in_rec (p_rec, start, end)) != NULL) |
| { |
| /* Check if attribute fits. Assume 3-byte value type/length */ |
| rem_len = *p_data_len - (UINT16) (p_rsp - p_data); |
| |
| if (p_attr->len > (UINT32)(rem_len - 6)) |
| break; |
| |
| p_rsp = sdpu_build_attrib_entry (p_rsp, p_attr); |
| |
| /* next attr id */ |
| start = p_attr->id + 1; |
| } |
| len = (INT32) (p_rsp - p_data); |
| |
| /* Put in the sequence header (2 or 3 bytes) */ |
| if (len > 255) |
| { |
| offset = 0; |
| p_data[0] = (UINT8) ((DATA_ELE_SEQ_DESC_TYPE << 3) | SIZE_IN_NEXT_WORD); |
| p_data[1] = (UINT8) ((len - 3) >> 8); |
| p_data[2] = (UINT8) (len - 3); |
| } |
| else |
| { |
| offset = 1; |
| |
| p_data[1] = (UINT8) ((DATA_ELE_SEQ_DESC_TYPE << 3) | SIZE_IN_NEXT_BYTE); |
| p_data[2] = (UINT8) (len - 3); |
| |
| len--; |
| } |
| *p_data_len = len; |
| } |
| #endif |
| /* If here, not found */ |
| return (offset); |
| } |
| #endif |
| |
| |
| |