| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992 |
- /******************************************************************************
- *
- * 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 "common/bt_target.h"
- #include "osi/allocator.h"
- #include "stack/l2cdefs.h"
- #include "stack/hcidefs.h"
- #include "stack/hcimsgs.h"
- #include "stack/sdp_api.h"
- #include "sdpint.h"
- #include "osi/list.h"
- #if (SDP_INCLUDED == TRUE)
- #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;
- list_node_t *p_node = NULL;
- /* If NULL, start at the beginning, else start at the first specified record */
- if (!p_rec) {
- p_node = list_begin(sdp_cb.server_db.p_record_list);
- } else {
- /* get node in the record list with given p_rec */
- p_node = list_get_node(sdp_cb.server_db.p_record_list, p_rec);
- if (p_node == NULL) {
- return NULL;
- }
- /* get next node */
- p_node = list_next(p_node);
- }
- /* Look through the records. The spec says that a match occurs if */
- /* the record contains all the passed UUIDs in it. */
- for( ; p_node; p_node = list_next(p_node)) {
- p_rec = list_node(p_node);
- 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;
- list_node_t *p_node = NULL;
- /* Look through the records for the caller's handle */
- for(p_node = list_begin(sdp_cb.server_db.p_record_list); p_node; p_node = list_next(p_node)) {
- p_rec = list_node(p_node);
- 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;
- tSDP_RECORD *p_rec = NULL;
- tSDP_RECORD *p_rec_prev = NULL;
- /* First, check if there is a free record */
- if (p_db->num_records < SDP_MAX_RECORDS) {
- p_rec =(tSDP_RECORD *)osi_malloc(sizeof(tSDP_RECORD));
- if (p_rec) {
- memset(p_rec, 0, sizeof(tSDP_RECORD));
- /* Save previous rec */
- if (p_db->num_records) {
- p_rec_prev = list_back(p_db->p_record_list);
- }
- /* Append new record */
- list_append(p_db->p_record_list, p_rec);
- /* We will use a handle of the first unreserved handle plus last record
- ** number + 1 */
- if (p_db->num_records) {
- handle = p_rec_prev->record_handle + 1;
- } else {
- handle = 0x10000;
- }
- p_rec->record_handle = handle;
- p_db->num_records++;
- SDP_TRACE_DEBUG("SDP_CreateRecord ok, num_records:%d\n", 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_rec->record_handle);
- } else {
- SDP_TRACE_ERROR("SDP_CreateRecord fail, memory allocation failed\n");
- }
- } else {
- SDP_TRACE_ERROR("SDP_CreateRecord fail, exceed maximum records:%d\n", 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
- tSDP_RECORD *p_rec = NULL;
- list_node_t *p_node = NULL;
- if (handle == 0 || sdp_cb.server_db.num_records == 0) {
- /* Delete all records in the database */
- sdp_cb.server_db.num_records = 0;
- for(p_node = list_begin(sdp_cb.server_db.p_record_list); p_node; p_node = list_next(p_node)) {
- list_remove(sdp_cb.server_db.p_record_list, p_node);
- }
- /* require new DI record to be created in SDP_SetLocalDiRecord */
- sdp_cb.server_db.di_primary_handle = 0;
- return (TRUE);
- } else {
- /* Find the record in the database */
- for(p_node = list_begin(sdp_cb.server_db.p_record_list); p_node; p_node = list_next(p_node)) {
- p_rec = list_node(p_node);
- if (p_rec->record_handle == handle) {
- /* Found it. Shift everything up one */
- list_remove(sdp_cb.server_db.p_record_list, p_rec);
- sdp_cb.server_db.num_records--;
- SDP_TRACE_DEBUG("SDP_DeleteRecord ok, num_records:%d\n", 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;
- }
- 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;
- tSDP_RECORD *p_rec = NULL;
- list_node_t *p_node= NULL;
- #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_DEBUG("SDP_AddAttribute: handle:%X, id:%04X, type:%d, len:%d, p_val:%p, *p_val:%s\n",
- handle, attr_id, attr_type, attr_len, p_val, num_array);
- } else if (attr_type == BOOLEAN_DESC_TYPE) {
- SDP_TRACE_DEBUG("SDP_AddAttribute: handle:%X, id:%04X, type:%d, len:%d, p_val:%p, *p_val:%d\n",
- handle, attr_id, attr_type, attr_len, p_val, *p_val);
- } else {
- SDP_TRACE_DEBUG("SDP_AddAttribute: handle:%X, id:%04X, type:%d, len:%d, p_val:%p, *p_val:%s\n",
- handle, attr_id, attr_type, attr_len, p_val, p_val);
- }
- }
- #endif
- /* Find the record in the database */
- for(p_node = list_begin(sdp_cb.server_db.p_record_list); p_node; p_node = list_next(p_node)) {
- p_rec= list_node(p_node);
- 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_WARNING("SDP_AddAttribute: attr_len:%d too long. truncate to (%d)\n",
- 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_ERROR("SDP_AddAttribute fail, length exceed maximum: ID %d: attr_len:%d \n",
- attr_id, attr_len );
- p_attr->id = p_attr->type = p_attr->len = 0;
- return (FALSE);
- }
- p_rec->num_attributes++;
- 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 *) osi_malloc(sizeof(UINT8) * SDP_MAX_ATTR_LEN * 2)) == NULL) {
- SDP_TRACE_ERROR("SDP_AddSequence cannot get a buffer!\n");
- 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_ERROR ("SDP_AddSequence - too long(attribute is not added)!!\n");
- osi_free(p_buff);
- return FALSE;
- } else {
- SDP_TRACE_ERROR ("SDP_AddSequence - too long, add %d elements of %d\n", xx, num_elem);
- }
- break;
- }
- }
- result = SDP_AddAttribute (handle, attr_id, DATA_ELE_SEQ_DESC_TYPE, (UINT32) (p - p_buff), p_buff);
- osi_free(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 *) osi_malloc(sizeof(UINT8) * SDP_MAX_ATTR_LEN * 2)) == NULL) {
- SDP_TRACE_ERROR("SDP_AddUuidSequence cannot get a buffer!\n");
- 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_WARNING ("SDP_AddUuidSequence - too long, add %d uuids of %d\n", xx, num_uuids);
- break;
- }
- }
- result = SDP_AddAttribute (handle, attr_id, DATA_ELE_SEQ_DESC_TYPE, (UINT32) (p - p_buff), p_buff);
- osi_free(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 *) osi_malloc(sizeof(UINT8) * SDP_MAX_ATTR_LEN * 2)) == NULL) {
- SDP_TRACE_ERROR("SDP_AddProtocolList cannot get a buffer!\n");
- 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);
- osi_free(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 *) osi_malloc(sizeof(UINT8) * SDP_MAX_ATTR_LEN * 2)) == NULL) {
- SDP_TRACE_ERROR("SDP_AddAdditionProtoLists cannot get a buffer!\n");
- 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);
- osi_free(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 *) osi_malloc(sizeof(UINT8) * SDP_MAX_ATTR_LEN)) == NULL) {
- SDP_TRACE_ERROR("SDP_AddProfileDescriptorList cannot get a buffer!\n");
- 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);
- osi_free(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 *) osi_malloc(sizeof(UINT8) * SDP_MAX_ATTR_LEN)) == NULL) {
- SDP_TRACE_ERROR("SDP_AddLanguageBaseAttrIDList cannot get a buffer!\n");
- 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);
- osi_free(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 *) osi_malloc(sizeof(UINT8) * SDP_MAX_ATTR_LEN * 2)) == NULL) {
- SDP_TRACE_ERROR("SDP_AddServiceClassIdList cannot get a buffer!\n");
- 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);
- osi_free(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 = NULL;
- list_node_t *p_node= NULL;
- UINT8 *pad_ptr;
- UINT32 len; /* Number of bytes in the entry */
- /* Find the record in the database */
- for(p_node = list_begin(sdp_cb.server_db.p_record_list); p_node; p_node = list_next(p_node)) {
- p_rec= list_node(p_node);
- if (p_rec->record_handle == handle) {
- tSDP_ATTRIBUTE *p_attr = &p_rec->attribute[0];
- SDP_TRACE_API("Deleting attr_id 0x%04x for handle 0x%x\n", 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 offset = -1; /* default to not found */
- #if SDP_SERVER_ENABLED == TRUE
- INT32 len = 0; /* Number of bytes in the entry */
- 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
- #endif ///SDP_INCLUDED == TRUE
|