📄 af.c
字号:
/*********************************************************************
Filename: AF.c
Revised: $Date: 2007-05-14 17:34:18 -0700 (Mon, 14 May 2007) $
Revision: $Revision: 14296 $
Description: General Operational Framework
- Device Description helper functions
Notes:
If supporting KVP, then afRegister() ties endpoint to KVP only in V2.
If KVP in V2,
Copyright (c) 2006 by Texas Instruments, Inc.
All Rights Reserved. Permission to use, reproduce, copy, prepare
derivative works, modify, distribute, perform, display or sell this
software and/or its documentation for any purpose is prohibited
without the express written consent of Texas Instruments, Inc.
*********************************************************************/
/*********************************************************************
* INCLUDES
*/
#include "OSAL.h"
#include "AF.h"
#include "nwk_globals.h"
#include "nwk_util.h"
#include "aps_groups.h"
#include "ZDProfile.h"
#include "aps_frag.h"
#if ( AF_FLOAT_SUPPORT )
#include "math.h"
#endif
#if defined ( MT_AF_CB_FUNC )
#include "MT_AF.h"
#endif
/*********************************************************************
* MACROS
*/
/*********************************************************************
* @fn afSend
*
* @brief Helper macro for V1 API to invoke V2 API.
*
* input parameters
*
* @param *dstAddr - Full ZB destination address: Nwk Addr + End Point.
* @param srcEP - Origination (i.e. respond to or ack to) End Point.
* @param cID - A valid cluster ID as specified by the Profile.
* @param len - Number of bytes of data pointed to by next param.
* @param *buf - A pointer to the data bytes to send.
* @param options - Valid bit mask of AF Tx Options as defined in AF.h.
* @param *transID - A pointer to a byte which can be modified and which will
* be used as the transaction sequence number of the msg.
*
* output parameters
*
* @param *transID - Incremented by one if the return value is success.
*
* @return afStatus_t - See previous definition of afStatus_... types.
*/
#if ( AF_V1_SUPPORT )
#define afSend( dstAddr, srcEP, cID, len, buf, transID, options, radius ) \
afDataRequest( (dstAddr), afFindEndPointDesc( (srcEP) ), \
(cID), (len), (buf), (transID), (options), (radius) )
#else
#define afSend( dstAddr, srcEP, cID, len, buf, transID, options, radius ) \
AF_DataRequest( (dstAddr), afFindEndPointDesc( (srcEP) ), \
(cID), (len), (buf), (transID), (options), (radius) )
#endif
/*********************************************************************
* CONSTANTS
*/
/*********************************************************************
* TYPEDEFS
*/
#if ( AF_KVP_SUPPORT )
typedef struct
{
uint16 clusterID;
byte transCount;
byte options;
byte srcEP;
afAddrType_t dstAddr;
uint16 msgLen;
byte *msg;
} afMultiHdr_t;
#endif
/*********************************************************************
* GLOBAL VARIABLES
*/
epList_t *epList;
/*********************************************************************
* EXTERNAL VARIABLES
*/
/*********************************************************************
* EXTERNAL FUNCTIONS
*/
/*********************************************************************
* LOCAL VARIABLES
*/
/*********************************************************************
* LOCAL FUNCTIONS
*/
#if ( AF_KVP_SUPPORT )
static void afBuildKVPIncoming( aps_FrameFormat_t *aff, endPointDesc_t *epDesc,
zAddrType_t *SrcAddress, uint8 LinkQuality, byte SecurityUse );
#endif
static void afBuildMSGIncoming( aps_FrameFormat_t *aff, endPointDesc_t *epDesc,
zAddrType_t *SrcAddress, uint8 LinkQuality, byte SecurityUse,
uint32 timestamp );
#if ( AF_KVP_SUPPORT )
static afMultiHdr_t *multiInit( afAddrType_t *dstAddr,
byte srcEndPoint, uint16 clusterID, byte FrameType,
byte txOptions, bool DiscoverRoute, byte RadiusCounter );
static bool multiAppend( afMultiHdr_t *hdr, byte bufLen, byte *buf,
byte CommandType, byte AttribDataType, uint16 AttribId, byte ErrorCode );
static afStatus_t multiSend( afMultiHdr_t *hdr, byte *seqNum, byte radius );
#endif
#if ( AF_V1_SUPPORT )
static afStatus_t afDataRequest( afAddrType_t *dstAddr, endPointDesc_t *srcEP,
uint16 cID, uint16 len, uint8 *buf, uint8 *transID,
uint8 options, uint8 radius );
#endif
static epList_t *afFindEndPointDescList( byte EndPoint );
static pDescCB afGetDescCB( endPointDesc_t *epDesc );
/*********************************************************************
* NETWORK LAYER CALLBACKS
*/
/*********************************************************************
* PUBLIC FUNCTIONS
*/
/*********************************************************************
* @fn afInit
*
* @brief Initialization function for the AF.
*
* @param none
*
* @return none
*/
void afInit( void )
{
// Start with no endpoint defined
epList = NULL;
}
/*********************************************************************
* @fn afRegisterExtended
*
* @brief Register an Application's EndPoint description.
*
* @param epDesc - pointer to the Application's endpoint descriptor.
* @param descFn - pointer to descriptor callback function
*
* NOTE: The memory that epDesc is pointing to must exist after this call.
*
* @return Pointer to epList_t on success, NULL otherwise.
*/
epList_t *afRegisterExtended( endPointDesc_t *epDesc, pDescCB descFn )
{
epList_t *ep;
epList_t *epSearch;
ep = osal_mem_alloc( sizeof ( epList_t ) );
if ( ep )
{
// Fill in the new list entry
ep->epDesc = epDesc;
#if !defined ( REFLECTOR )
ep->reflectorAddr = NWK_PAN_COORD_ADDR;
#else
ep->reflectorAddr = INVALID_NODE_ADDR;
#endif
// Default to allow Match Descriptor.
ep->flags = eEP_AllowMatch;
#if ( AF_KVP_SUPPORT )
ep->flags |= ((epDesc->endPoint == ZDO_EP) ? 0 : eEP_UsesKVP);
#endif
ep->pfnDescCB = descFn;
ep->nextDesc = NULL;
// Does a list exist?
if ( epList == NULL )
epList = ep; // Make this the first entry
else
{
// Look for the end of the list
epSearch = epList;
while( epSearch->nextDesc != NULL )
epSearch = epSearch->nextDesc;
// Add new entry to end of list
epSearch->nextDesc = ep;
}
}
return ep;
}
/*********************************************************************
* @fn afRegister
*
* @brief Register an Application's EndPoint description.
*
* @param epDesc - pointer to the Application's endpoint descriptor.
*
* NOTE: The memory that epDesc is pointing to must exist after this call.
*
* @return afStatus_SUCCESS - Registered
* afStatus_MEM_FAIL - not enough memory to add descriptor
*/
afStatus_t afRegister( endPointDesc_t *epDesc )
{
epList_t *ep = afRegisterExtended( epDesc, NULL );
return ((ep == NULL) ? afStatus_MEM_FAIL : afStatus_SUCCESS);
}
#if ( AF_KVP_SUPPORT )
/*********************************************************************
* @fn afRegisterFlags
*
* @brief Register an Application's EndPoint description.
*
* @param epDesc - pointer to the Application's endpoint descriptor.
*
* NOTE: The memory that epDesc is pointing to must exist after this call.
*
* @return afStatus_SUCCESS - Registered
* afStatus_MEM_FAIL - not enough memory to add descriptor
*/
afStatus_t afRegisterFlags( endPointDesc_t *epDesc, eEP_Flags flags )
{
epList_t *ep = afRegisterExtended( epDesc, NULL );
if ( ep != NULL )
{
ep->flags = flags;
return afStatus_SUCCESS;
}
else
{
return afStatus_MEM_FAIL;
}
}
#endif
/*********************************************************************
* @fn afDataConfirm
*
* @brief This function will generate the Data Confirm back to
* the application.
*
* @param endPoint - confirm end point
* @param transID - transaction ID from APSDE_DATA_REQUEST
* @param status - status of APSDE_DATA_REQUEST
*
* @return none
*/
void afDataConfirm( uint8 endPoint, uint8 transID, ZStatus_t status )
{
endPointDesc_t *epDesc;
afDataConfirm_t *msgPtr;
// Find the endpoint description
epDesc = afFindEndPointDesc( endPoint );
if ( epDesc == NULL )
return;
// Determine the incoming command type
msgPtr = (afDataConfirm_t *)osal_msg_allocate( sizeof(afDataConfirm_t) );
if ( msgPtr )
{
// Build the Data Confirm message
msgPtr->hdr.event = AF_DATA_CONFIRM_CMD;
msgPtr->hdr.status = status;
msgPtr->endpoint = endPoint;
#if ( AF_V1_SUPPORT )
msgPtr->transID = transID;
#elif ( AF_KVP_SUPPORT )
{
epList_t *pList = afFindEndPointDescList( endPoint );
if ( pList )
{
if ( pList->flags & eEP_UsesKVP )
{
msgPtr->transID = transID;
}
}
}
#else
(void)transID;
#endif
// send message through task message
osal_msg_send( *(epDesc->task_id), (byte *)msgPtr );
}
}
/*********************************************************************
* @fn afIncomingData
*
* @brief Transfer a data PDU (ASDU) from the APS sub-layer to the AF.
*
* @param aff - pointer to APS frame format
* @param SrcAddress - Source address
* @param LinkQuality - incoming message's link quality
* @param SecurityUse - Security enable/disable
*
* @return none
*/
void afIncomingData( aps_FrameFormat_t *aff, zAddrType_t *SrcAddress,
uint8 LinkQuality, byte SecurityUse, uint32 timestamp )
{
endPointDesc_t *epDesc = NULL;
uint16 epProfileID = 0xFFFF; // Invalid Profile ID
epList_t *pList;
uint8 grpEp;
if ( (NLME_GetProtocolVersion() != ZB_PROT_V1_0)
&& ((aff->FrmCtrl & APS_DELIVERYMODE_MASK) == APS_FC_DM_GROUP) )
{
// Find the first endpoint for this group
grpEp = aps_FindGroupForEndpoint( aff->GroupID, APS_GROUPS_FIND_FIRST );
if ( grpEp == APS_GROUPS_EP_NOT_FOUND )
return; // No endpoint found
epDesc = afFindEndPointDesc( grpEp );
if ( epDesc == NULL )
return; // Endpoint descriptor not found
pList = afFindEndPointDescList( epDesc->endPoint );
}
else if ( aff->DstEndPoint == AF_BROADCAST_ENDPOINT )
{
// Set the list
if ( (pList = epList) )
{
epDesc = pList->epDesc;
}
}
else if ( (epDesc = afFindEndPointDesc( aff->DstEndPoint )) )
{
pList = afFindEndPointDescList( epDesc->endPoint );
}
while ( epDesc )
{
if ( pList->pfnDescCB )
{
uint16 *pID = (uint16 *)(pList->pfnDescCB(
AF_DESCRIPTOR_PROFILE_ID, epDesc->endPoint ));
if ( pID )
{
epProfileID = *pID;
osal_mem_free( pID );
}
}
else if ( epDesc->simpleDesc )
{
epProfileID = epDesc->simpleDesc->AppProfId;
}
if ( (aff->ProfileID == epProfileID) ||
((epDesc->endPoint == ZDO_EP) && (aff->ProfileID == ZDO_PROFILE_ID)) )
{
#if ( AF_KVP_SUPPORT )
if ( (FRAMETYPE_KVP == HI_UINT8(aff->asdu[0])) &&
(pList->flags & eEP_UsesKVP) )
{
afBuildKVPIncoming( aff, epDesc, SrcAddress, LinkQuality, SecurityUse );
}
else
#endif
{
afBuildMSGIncoming( aff, epDesc, SrcAddress, LinkQuality, SecurityUse, timestamp );
}
}
if ( (NLME_GetProtocolVersion() != ZB_PROT_V1_0)
&& ((aff->FrmCtrl & APS_DELIVERYMODE_MASK) == APS_FC_DM_GROUP) )
{
// Find the next endpoint for this group
grpEp = aps_FindGroupForEndpoint( aff->GroupID, grpEp );
if ( grpEp == APS_GROUPS_EP_NOT_FOUND )
return; // No endpoint found
epDesc = afFindEndPointDesc( grpEp );
if ( epDesc == NULL )
return; // Endpoint descriptor not found
pList = afFindEndPointDescList( epDesc->endPoint );
}
else if ( aff->DstEndPoint == AF_BROADCAST_ENDPOINT )
{
pList = pList->nextDesc;
if ( pList )
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -