/******************************************************************************
*
* Copyright (C) 2010-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.
*
******************************************************************************/
/******************************************************************************
*
* This file contains the LLCP Service Discovery
*
******************************************************************************/
#include <string.h>
#include "gki.h"
#include "nfc_target.h"
#include "bt_types.h"
#include "llcp_api.h"
#include "llcp_int.h"
#include "llcp_defs.h"
/*******************************************************************************
**
** Function llcp_sdp_proc_data
**
** Description Do nothing
**
**
** Returns void
**
*******************************************************************************/
void llcp_sdp_proc_data (tLLCP_SAP_CBACK_DATA *p_data)
{
/*
** Do nothing
** llcp_sdp_proc_SNL () is called by link layer
*/
}
/*******************************************************************************
**
** Function llcp_sdp_check_send_snl
**
** Description Enqueue Service Name Lookup PDU into sig_xmit_q for transmitting
**
**
** Returns void
**
*******************************************************************************/
void llcp_sdp_check_send_snl (void)
{
UINT8 *p;
if (llcp_cb.sdp_cb.p_snl)
{
LLCP_TRACE_DEBUG0 ("SDP: llcp_sdp_check_send_snl ()");
llcp_cb.sdp_cb.p_snl->len += LLCP_PDU_HEADER_SIZE;
llcp_cb.sdp_cb.p_snl->offset -= LLCP_PDU_HEADER_SIZE;
p = (UINT8 *) (llcp_cb.sdp_cb.p_snl + 1) + llcp_cb.sdp_cb.p_snl->offset;
UINT16_TO_BE_STREAM (p, LLCP_GET_PDU_HEADER (LLCP_SAP_SDP, LLCP_PDU_SNL_TYPE, LLCP_SAP_SDP ));
GKI_enqueue (&llcp_cb.lcb.sig_xmit_q, llcp_cb.sdp_cb.p_snl);
llcp_cb.sdp_cb.p_snl = NULL;
}
else
{
/* Notify DTA after sending out SNL with SDRES not to send SNLs in AGF PDU */
if ((llcp_cb.p_dta_cback) && (llcp_cb.dta_snl_resp))
{
llcp_cb.dta_snl_resp = FALSE;
(*llcp_cb.p_dta_cback) ();
}
}
}
/*******************************************************************************
**
** Function llcp_sdp_add_sdreq
**
** Description Add Service Discovery Request into SNL PDU
**
**
** Returns void
**
*******************************************************************************/
static void llcp_sdp_add_sdreq (UINT8 tid, char *p_name)
{
UINT8 *p;
UINT16 name_len = (UINT16) strlen (p_name);
p = (UINT8 *) (llcp_cb.sdp_cb.p_snl + 1) + llcp_cb.sdp_cb.p_snl->offset + llcp_cb.sdp_cb.p_snl->len;
UINT8_TO_BE_STREAM (p, LLCP_SDREQ_TYPE);
UINT8_TO_BE_STREAM (p, (1 + name_len));
UINT8_TO_BE_STREAM (p, tid);
ARRAY_TO_BE_STREAM (p, p_name, name_len);
llcp_cb.sdp_cb.p_snl->len += LLCP_SDREQ_MIN_LEN + name_len;
}
/*******************************************************************************
**
** Function llcp_sdp_send_sdreq
**
** Description Send Service Discovery Request
**
**
** Returns LLCP_STATUS
**
*******************************************************************************/
tLLCP_STATUS llcp_sdp_send_sdreq (UINT8 tid, char *p_name)
{
tLLCP_STATUS status;
UINT16 name_len;
UINT16 available_bytes;
LLCP_TRACE_DEBUG2 ("llcp_sdp_send_sdreq (): tid=0x%x, ServiceName=%s", tid, p_name);
/* if there is no pending SNL */
if (!llcp_cb.sdp_cb.p_snl)
{
llcp_cb.sdp_cb.p_snl = (BT_HDR*) GKI_getpoolbuf (LLCP_POOL_ID);
if (llcp_cb.sdp_cb.p_snl)
{
llcp_cb.sdp_cb.p_snl->offset = NCI_MSG_OFFSET_SIZE + NCI_DATA_HDR_SIZE + LLCP_PDU_HEADER_SIZE;
llcp_cb.sdp_cb.p_snl->len = 0;
}
}
if (llcp_cb.sdp_cb.p_snl)
{
available_bytes = GKI_get_buf_size (llcp_cb.sdp_cb.p_snl)
- BT_HDR_SIZE - llcp_cb.sdp_cb.p_snl->offset
- llcp_cb.sdp_cb.p_snl->len;
name_len = (UINT16) strlen (p_name);
/* if SDREQ parameter can be added in SNL */
if ( (available_bytes >= LLCP_SDREQ_MIN_LEN + name_len)
&&(llcp_cb.sdp_cb.p_snl->len + LLCP_SDREQ_MIN_LEN + name_len <= llcp_cb.lcb.effective_miu) )
{
llcp_sdp_add_sdreq (tid, p_name);
status = LLCP_STATUS_SUCCESS;
}
else
{
/* send pending SNL PDU to LM */
llcp_sdp_check_send_snl ();
llcp_cb.sdp_cb.p_snl = (BT_HDR*) GKI_getpoolbuf (LLCP_POOL_ID);
if (llcp_cb.sdp_cb.p_snl)
{
llcp_cb.sdp_cb.p_snl->offset = NCI_MSG_OFFSET_SIZE + NCI_DATA_HDR_SIZE + LLCP_PDU_HEADER_SIZE;
llcp_cb.sdp_cb.p_snl->len = 0;
llcp_sdp_add_sdreq (tid, p_name);
status = LLCP_STATUS_SUCCESS;
}
else
{
status = LLCP_STATUS_FAIL;
}
}
}
else
{
status = LLCP_STATUS_FAIL;
}
/* if LM is waiting for PDUs from upper layer */
if ( (status == LLCP_STATUS_SUCCESS)
&&(llcp_cb.lcb.symm_state == LLCP_LINK_SYMM_LOCAL_XMIT_NEXT) )
{
llcp_link_check_send_data ();
}
return status;
}
/*******************************************************************************
**
** Function llcp_sdp_add_sdres
**
** Description Add Service Discovery Response into SNL PDU
**
**
** Returns void
**
*******************************************************************************/
static void llcp_sdp_add_sdres (UINT8 tid, UINT8 sap)
{
UINT8 *p;
p = (UINT8 *) (llcp_cb.sdp_cb.p_snl + 1) + llcp_cb.sdp_cb.p_snl->offset + llcp_cb.sdp_cb.p_snl->len;
UINT8_TO_BE_STREAM (p, LLCP_SDRES_TYPE);
UINT8_TO_BE_STREAM (p, LLCP_SDRES_LEN);
UINT8_TO_BE_STREAM (p, tid);
UINT8_TO_BE_STREAM (p, sap);
llcp_cb.sdp_cb.p_snl->len += 2 + LLCP_SDRES_LEN; /* type and length */
}
/*******************************************************************************
**
** Function llcp_sdp_send_sdres
**
** Description Send Service Discovery Response
**
**
** Returns LLCP_STATUS
**
*******************************************************************************/
static tLLCP_STATUS llcp_sdp_send_sdres (UINT8 tid, UINT8 sap)
{
tLLCP_STATUS status;
UINT16 available_bytes;
LLCP_TRACE_DEBUG2 ("llcp_sdp_send_sdres (): tid=0x%x, SAP=0x%x", tid, sap);
/* if there is no pending SNL */
if (!llcp_cb.sdp_cb.p_snl)
{
llcp_cb.sdp_cb.p_snl = (BT_HDR*) GKI_getpoolbuf (LLCP_POOL_ID);
if (llcp_cb.sdp_cb.p_snl)
{
llcp_cb.sdp_cb.p_snl->offset = NCI_MSG_OFFSET_SIZE + NCI_DATA_HDR_SIZE + LLCP_PDU_HEADER_SIZE;
llcp_cb.sdp_cb.p_snl->len = 0;
}
}
if (llcp_cb.sdp_cb.p_snl)
{
available_bytes = GKI_get_buf_size (llcp_cb.sdp_cb.p_snl)
- BT_HDR_SIZE - llcp_cb.sdp_cb.p_snl->offset
- llcp_cb.sdp_cb.p_snl->len;
/* if SDRES parameter can be added in SNL */
if ( (available_bytes >= 2 + LLCP_SDRES_LEN)
&&(llcp_cb.sdp_cb.p_snl->len + 2 + LLCP_SDRES_LEN <= llcp_cb.lcb.effective_miu) )
{
llcp_sdp_add_sdres (tid, sap);
status = LLCP_STATUS_SUCCESS;
}
else
{
/* send pending SNL PDU to LM */
llcp_sdp_check_send_snl ();
llcp_cb.sdp_cb.p_snl = (BT_HDR*) GKI_getpoolbuf (LLCP_POOL_ID);
if (llcp_cb.sdp_cb.p_snl)
{
llcp_cb.sdp_cb.p_snl->offset = NCI_MSG_OFFSET_SIZE + NCI_DATA_HDR_SIZE + LLCP_PDU_HEADER_SIZE;
llcp_cb.sdp_cb.p_snl->len = 0;
llcp_sdp_add_sdres (tid, sap);
status = LLCP_STATUS_SUCCESS;
}
else
{
status = LLCP_STATUS_FAIL;
}
}
}
else
{
status = LLCP_STATUS_FAIL;
}
/* if LM is waiting for PDUs from upper layer */
if ( (status == LLCP_STATUS_SUCCESS)
&&(llcp_cb.lcb.symm_state == LLCP_LINK_SYMM_LOCAL_XMIT_NEXT) )
{
llcp_link_check_send_data ();
}
return status;
}
/*******************************************************************************
**
** Function llcp_sdp_get_sap_by_name
**
** Description Search SAP by service name
**
**
** Returns SAP if success
**
*******************************************************************************/
UINT8 llcp_sdp_get_sap_by_name (char *p_name, UINT8 length)
{
UINT8 sap;
tLLCP_APP_CB *p_app_cb;
for (sap = LLCP_SAP_SDP; sap <= LLCP_UPPER_BOUND_SDP_SAP; sap++)
{
p_app_cb = llcp_util_get_app_cb (sap);
if ( (p_app_cb)
&&(p_app_cb->p_app_cback)
&&(strlen((char*)p_app_cb->p_service_name) == length)
&&(!strncmp((char*)p_app_cb->p_service_name, p_name, length)) )
{
/* if device is under LLCP DTA testing */
if ( (llcp_cb.p_dta_cback)
&&(!strncmp((char*)p_app_cb->p_service_name, "urn:nfc:sn:cl-echo-in", length)) )
{
llcp_cb.dta_snl_resp = TRUE;
}
return (sap);
}
}
return 0;
}
/*******************************************************************************
**
** Function llcp_sdp_return_sap
**
** Description Report TID and SAP to requester
**
**
** Returns void
**
*******************************************************************************/
static void llcp_sdp_return_sap (UINT8 tid, UINT8 sap)
{
UINT8 i;
LLCP_TRACE_DEBUG2 ("llcp_sdp_return_sap (): tid=0x%x, SAP=0x%x", tid, sap);
for (i = 0; i < LLCP_MAX_SDP_TRANSAC; i++)
{
if ( (llcp_cb.sdp_cb.transac[i].p_cback)
&&(llcp_cb.sdp_cb.transac[i].tid == tid) )
{
(*llcp_cb.sdp_cb.transac[i].p_cback) (tid, sap);
llcp_cb.sdp_cb.transac[i].p_cback = NULL;
}
}
}
/*******************************************************************************
**
** Function llcp_sdp_proc_deactivation
**
** Description Report SDP failure for any pending request because of deactivation
**
**
** Returns void
**
*******************************************************************************/
void llcp_sdp_proc_deactivation (void)
{
UINT8 i;
LLCP_TRACE_DEBUG0 ("llcp_sdp_proc_deactivation ()");
for (i = 0; i < LLCP_MAX_SDP_TRANSAC; i++)
{
if (llcp_cb.sdp_cb.transac[i].p_cback)
{
(*llcp_cb.sdp_cb.transac[i].p_cback) (llcp_cb.sdp_cb.transac[i].tid, 0x00);
llcp_cb.sdp_cb.transac[i].p_cback = NULL;
}
}
/* free any pending SNL PDU */
if (llcp_cb.sdp_cb.p_snl)
{
GKI_freebuf (llcp_cb.sdp_cb.p_snl);
llcp_cb.sdp_cb.p_snl = NULL;
}
llcp_cb.sdp_cb.next_tid = 0;
llcp_cb.dta_snl_resp = FALSE;
}
/*******************************************************************************
**
** Function llcp_sdp_proc_snl
**
** Description Process SDREQ and SDRES in SNL
**
**
** Returns LLCP_STATUS
**
*******************************************************************************/
tLLCP_STATUS llcp_sdp_proc_snl (UINT16 sdu_length, UINT8 *p)
{
UINT8 type, length, tid, sap, *p_value;
LLCP_TRACE_DEBUG0 ("llcp_sdp_proc_snl ()");
if ((llcp_cb.lcb.agreed_major_version < LLCP_MIN_SNL_MAJOR_VERSION)||
((llcp_cb.lcb.agreed_major_version == LLCP_MIN_SNL_MAJOR_VERSION)&&(llcp_cb.lcb.agreed_minor_version < LLCP_MIN_SNL_MINOR_VERSION)))
{
LLCP_TRACE_DEBUG0 ("llcp_sdp_proc_snl(): version number less than 1.1, SNL not supported.");
return LLCP_STATUS_FAIL;
}
while (sdu_length >= 2) /* at least type and length */
{
BE_STREAM_TO_UINT8 (type, p);
BE_STREAM_TO_UINT8 (length, p);
switch (type)
{
case LLCP_SDREQ_TYPE:
if ( (length > 1) /* TID and sevice name */
&&(sdu_length >= 2 + length) ) /* type, length, TID and service name */
{
p_value = p;
BE_STREAM_TO_UINT8 (tid, p_value);
sap = llcp_sdp_get_sap_by_name ((char*) p_value, (UINT8) (length - 1));
llcp_sdp_send_sdres (tid, sap);
}
else
{
LLCP_TRACE_ERROR1 ("llcp_sdp_proc_snl (): bad length (%d) in LLCP_SDREQ_TYPE", length);
}
break;
case LLCP_SDRES_TYPE:
if ( (length == LLCP_SDRES_LEN) /* TID and SAP */
&&(sdu_length >= 2 + length) ) /* type, length, TID and SAP */
{
p_value = p;
BE_STREAM_TO_UINT8 (tid, p_value);
BE_STREAM_TO_UINT8 (sap, p_value);
llcp_sdp_return_sap (tid, sap);
}
else
{
LLCP_TRACE_ERROR1 ("llcp_sdp_proc_snl (): bad length (%d) in LLCP_SDRES_TYPE", length);
}
break;
default:
LLCP_TRACE_WARNING1 ("llcp_sdp_proc_snl (): Unknown type (0x%x) is ignored", type);
break;
}
if (sdu_length >= 2 + length) /* type, length, value */
{
sdu_length -= 2 + length;
p += length;
}
else
{
break;
}
}
if (sdu_length)
{
LLCP_TRACE_ERROR0 ("llcp_sdp_proc_snl (): Bad format of SNL");
return LLCP_STATUS_FAIL;
}
else
{
return LLCP_STATUS_SUCCESS;
}
}