admin 管理员组

文章数量: 1184232


2024年3月14日发(作者:serve as)

协议栈无线透传编程原理:

第一个功能:协调器的组网,终端设备和路由设备发现网络以及加入网络

//第一步:Z-Stack 由 main()函数开始执行,main()函数共做了 2 件

事:一是系统初始化,另外一件是开始执行轮转查询式操作系统

int main( void ) {

.......

// Initialize the operating system

osal_init_system(); //第二步,操作系统初始化

......

osal_start_system(); //初始化完系统任务事件后,正式开始执行操

作系统

......

}

//第二步,进入 osal_init_system()函数,执行操作系统初始化

uint8 osal_init_system( void ) //初始化操作系统,其中最重要的

是,初始化操作系统的任务

{

// Initialize the Memory Allocation System

osal_mem_init();

// Initialize the message queue

osal_qHead = NULL;

// Initialize the timers

osalTimerInit();

// Initialize the Power Management System

osal_pwrmgr_init();

// Initialize the system tasks.

osalInitTasks(); //第三步,执行操作系统

任务初始化函数

// Setup efficient search for the first free block of heap.

osal_mem_kick();

return ( SUCCESS );

}

//第三步,进入osalInitTasks()函数,执行操作系统任务初始化

void osalInitTasks( void ) //第三步,初始化操作系统任务

{

uint8 taskID = 0;

tasksEvents = (uint16 *)osal_mem_alloc( sizeof( uint16 ) * tasksCnt);

osal_memset( tasksEvents, 0, (sizeof( uint16 ) * tasksCnt));

//任务优先级由高向低依次排列,高优先级对应 taskID 的值反而小

macTaskInit( taskID++ ); //不需要用户考虑

nwk_init( taskID++ ); //不需要用户考虑

Hal_Init( taskID++ ); //硬件抽象层初始化,需要我们考虑

#if defined( MT_TASK )

MT_TaskInit( taskID++ );

#endif

APS_Init( taskID++ ); //不需要用户考虑

#if defined ( ZIGBEE_FRAGMENTATION )

APSF_Init( taskID++ );

#endif

ZDApp_Init( taskID++ ); //第四步,ZDApp层,初始化 ,执行

ZDApp_init函数后,如果是协调器将建立网络,如果是终端设备将加入网络。

#if defined ( ZIGBEE_FREQ_AGILITY ) || defined

( ZIGBEE_PANID_CONFLICT )

ZDNwkMgr_Init( taskID++ );

#endif

SerialApp_Init( taskID ); //应用层SerialApp层初始化,需要用户

考虑 在此处设置了一个按键触发事件,

//当有按键按下的时候,产生一个系统消息

}

//第四步,进入ZDApp_init()函数,执行ZDApp层初始化

//The first step

void ZDApp_Init( uint8 task_id ) //The first step,ZDApp层初始

化。

{

// Save the task ID

ZDAppTaskID = task_id;

// Initialize the ZDO global device short address storage

de = Addr16Bit;

ddr = INVALID_NODE_ADDR;

(void)NLME_GetExtAddr(); // Load the saveExtAddr pointer.

// Check for manual "Hold Auto Start"

ZDAppCheckForHoldKey();

// Initialize ZDO items and setup the device - type of device to create.

ZDO_Init();

// Register the endpoint description with the AF

// This task doesn't have a Simple description, but we still need

// to register the endpoint.

afRegister( (endPointDesc_t *)&ZDApp_epDesc );

#if defined( ZDO_USERDESC_RESPONSE )

ZDApp_InitUserDesc();

#endif // ZDO_USERDESC_RESPONSE

// Start the device?

if ( devState != DEV_HOLD ) //devState 初值为

DEV_INIT , 所以在初始化ZDA层时,就执行该条件语句

{

ZDOInitDevice( 0 ); //The second step, 接着转到

ZDOInitDevice()函数,执行The third step;

}

else

{

// Blink LED to indicate HOLD_START

HalLedBlink ( HAL_LED_4, 0, 50, 500 );

}

ZDApp_RegisterCBs();

} /* ZDApp_Init() */

//The third step,执行ZDOInitDevice()函数,执行设备初始化

uint8 ZDOInitDevice( uint16 startDelay ) //The third step, ZDO层

初始化设备,

{

.......

// Trigger the network start

ZDApp_NetworkInit( extendedDelay ); //网络初始化,跳到相应的函

数里头,执行The fourth step

.......

}

//The fouth step,执行 ZDApp_NetWorkInit()函数

void ZDApp_NetworkInit( uint16 delay ) //The fourth step,网络初始化

{

if ( delay )

{

// Wait awhile before starting the device

osal_start_timerEx( ZDAppTaskID, ZDO_NETWORK_INIT,

delay ); //发送ZDO_NETWORK_INIT(网络初始化)消息到 ZDApp层,转

//ZDApp层,执

行The fifth step , ZDApp_event_loop() 函数

}

else

{

osal_set_event( ZDAppTaskID, ZDO_NETWORK_INIT );

}

}

//The fifth step,转到ZDApp_event_loop()函数

UINT16 ZDApp_event_loop( uint8 task_id, UINT16 events )

{

if ( events & ZDO_NETWORK_INIT ) //The fivth step,网络初始化事件

处理

{

// Initialize apps and start the network

devState = DEV_INIT;

//设备逻辑类型,启动模式,信标时间,超帧长度,接着转到The sixth

step,去启动设备,接着执行The sixth step,转到ZDO_StartDevice()

ZDO_StartDevice( (uint8)ZDO_Config_Node_lType,

devStartMode,

DEFAULT_BEACON_ORDER,

DEFAULT_SUPERFRAME_ORDER );

// Return unprocessed events

return (events ^ ZDO_NETWORK_INIT);

}

}

//The sixth step,执行ZDO_StartDevice()函数,启动设备

void ZDO_StartDevice( byte logicalType, devStartModes_t startMode, byte

beaconOrder, byte superframeOrder ) //The sixth step

{

......

if ( ZG_BUILD_COORDINATOR_TYPE && logicalType ==

NODETYPE_COORDINATOR ) //当设备作为协调器时,执行这个条件语句。

{

if ( startMode == MODE_HARD )

{

devState = DEV_COORD_STARTING;

//向网络层发送网络形成请求。当网络层执行

NLME_NetworkFormationRequest()建立网络后,将给予 ZDO层反馈信息。

// 接着转到The seventh step,去执行ZDApp层

的 ZDO_NetworkFormationConfirmCB()函数

ret = NLME_NetworkFormationRequest( zgConfigPANID,

zgApsUseExtendedPANID, zgDefaultChannelList,

zgDefaultStartingScanDuration, beaconOrder,

superframeOrder, false );

}

if ( ZG_BUILD_JOINING_TYPE && (logicalType == NODETYPE_ROUTER ||

logicalType == NODETYPE_DEVICE) ) //当为终端设备或路由时

{

if ( (startMode == MODE_JOIN) || (startMode == MODE_REJOIN) )

{

devState = DEV_NWK_DISC;

// zgDefaultChannelList与协调器形成网络的通道号匹配。 网络

发现请求。

// 继而转到ZDO_NetworkDiscoveryConfirmCB()函数

ret = NLME_NetworkDiscoveryRequest( zgDefaultChannelList,

zgDefaultStartingScanDuration );

}

}

......

}

//The seventh step,分两种情况,1.协调器 2.路由器或终端设备

1)协调器

void ZDO_NetworkFormationConfirmCB( ZStatus_t Status ) //The seventh

step,给予ZDO层网络形成反馈信息(协调器)

{

osal_set_event( ZDAppTaskID, ZDO_NETWORK_START ); //发送网络启动事件 到

ZDApp层,接着转到ZDApp_event_loop()函

......

}

UINT16 ZDApp_event_loop( uint8 task_id, UINT16 events )

{

......

if ( events & ZDO_NETWORK_START ) // 网络启动事件

{

ZDApp_NetworkStartEvt(); //网络启动事件,接着跳转到The

ninth step, 执行ZDApp_NetworkStartEvt()函数

......

}

}

void ZDApp_NetworkStartEvt( void ) //处理网络启动事件

{

......

osal_pwrmgr_device( PWRMGR_ALWAYS_ON );

//电源总是上电

osal_set_event( ZDAppTaskID, ZDO_STATE_CHANGE_EVT ); //设置网络状态改

变事件,发送到ZDApp层,转到The tenth step,去

......

// ZDApp_event_loop()函数,找到相对应的

网络改变事件。

}

2)路由器或终端设备

//The seventh step(终端设备), 当发现有网络存在时,网络层将给予 ZDO 层

发现网络反馈信息

ZStatus_t ZDO_NetworkDiscoveryConfirmCB( uint8 ResultCount,

networkDesc_t *NetworkList )

{

.......

//把网络发现这个反馈消息,发送到ZDA层,转到 ZDApp_ProcessOSALMsg

(),执行 ZDApp_SendMsg( ZDAppTaskID, ZDO_NWK_DISC_CNF,

sizeof(ZDO_NetworkDiscoveryCfm_t), (uint8 *)&msg );

}

void ZDApp_ProcessOSALMsg( osal_event_hdr_t *msgPtr )

{

......

case ZDO_NWK_DISC_CNF: // (终端设备),网络发现响

应。 ......

//当发现有网络存在时,网络层将给予 ZDO 层发现网络反馈

信息。然后由网络层发起加入网络请求,

//如加入网络成功,则网络层将给予 ZDO 层加入网络反馈,

执行NLME_JoinRequest()函数。然后转到

//The ninth step,执行 ZDO_JoinConfirmCB()函数

if ( NLME_JoinRequest( ((ZDO_NetworkDiscoveryCfm_t

*)msgPtr)->extendedPANID,

BUILD_UINT16( ((ZDO_NetworkDiscoveryCfm_

t *)msgPtr)->panIdLSB, ((ZDO_NetworkDiscoveryCfm_t

*)msgPtr)->panIdMSB ),

((ZDO_NetworkDiscoveryCfm_t

*)msgPtr)->logicalChannel,

ZDO_Config_Node_lityFla

gs ) != ZSuccess )

{

ZDApp_NetworkInit( (uint16)(NWK_START_DELAY

+ ((uint16)(osal_rand()&

EXTENDED_JOINING_RANDOM_MASK))) );

}

......

}

void ZDO_JoinConfirmCB( uint16 PanId, ZStatus_t Status ) //The ninth

step(终端设备), 终端设备加入网络响应。

{

......

//将ZDO_NWK_JOIN_IND事件发送到ZDA层,执行 ZDApp_ProcessOSALMsg()函

数。

ZDApp_SendMsg( ZDAppTaskID, ZDO_NWK_JOIN_IND,

sizeof(osal_event_hdr_t), (byte*)NULL );

}

void ZDApp_ProcessOSALMsg( osal_event_hdr_t *msgPtr )

{

......

case ZDO_NWK_JOIN_IND: //终端设备,加入网络反馈信息

事件。

if ( ZG_BUILD_JOINING_TYPE && ZG_DEVICE_JOINING_TYPE )

{

ZDApp_ProcessNetworkJoin(); //转到

ZDApp_ProcessNetworkJoin(),执行ZDApp_ProcessNetworkJoin()函数。

}

break;

......

}

在执行ZDApp_ProcessNetworkJoin()函数的时候,要分两种情况,一种是终端

设备,一种是路由器:

3)终端设备:

void ZDApp_ProcessNetworkJoin( void ) //处理网络加入事件。

{

......

if ( nwkStatus == ZSuccess )

{

//设置 ZDO_STATE_CHANGE_EVT ,发送到ZDA层,执行

ZDApp_event_loop()函数。

osal_set_event( ZDAppTaskID, ZDO_STATE_CHANGE_EVT );

}

......

}

4)路由器:

void ZDApp_ProcessNetworkJoin( void )

{

......

if ( ZSTACK_ROUTER_BUILD )

{

// NOTE: first two parameters are not used, see NLMEDE.h

for details

if ( ZDO_Config_Node_lType !=

NODETYPE_DEVICE )

{

NLME_StartRouterRequest( 0, 0,

false ); //路由启动请求

}

}

......

}

void ZDO_StartRouterConfirmCB( ZStatus_t Status )

{

nwkStatus = (byte)Status;

......

osal_set_event( ZDAppTaskID, ZDO_ROUTER_START );

}

UINT16 ZDApp_event_loop( uint8 task_id, UINT16 events )

{

if ( events & ZDO_ROUTER_START )

{

if ( nwkStatus == ZSuccess )

{

if ( devState == DEV_END_DEVICE )

devState =

DEV_ROUTER; //设备状态变成路

由器

osal_pwrmgr_device( PWRMGR_ALWAYS_ON );

}

else

{

// remain as end device!!

}

osal_set_event( ZDAppTaskID,

ZDO_STATE_CHANGE_EVT ); //设置ZDO状态改变事件

// Return unprocessed events

return (events ^ ZDO_ROUTER_START);

}

}

//The eighth step,执行ZDO状态改变事件

UINT16 ZDApp_event_loop( uint8 task_id, UINT16 events )

{

.......

if ( events & ZDO_STATE_CHANGE_EVT ) //The eighth step, 网络改变事

件,这个事件就是在设备加入网络成功后,

//并在网络中的身份确定

后产生的一个事件

{

ZDO_UpdateNwkStatus( devState ); //更新网络状态,转到The

eleventh step,执行 ZDO_UpdateNwkStatus()函数。

......

}

}

//The ninth step,执行ZDO_UpdateNwkStatus()函数,完成网络状态更新

void ZDO_UpdateNwkStatus(devStates_t state) //The ninth step, 更新

网络状态

{

......

zdoSendStateChangeMsg(state,

*(pItem->epDesc->task_id)); //发送状态改变消息到zdo层,这是The

tenth step,转到

//zdoSendStateChangeMsg()函数

.......

ddr = NLME_GetShortAddr(); //调用

NLME_GetShortAddr()函数,获得16位短地址。

(void)NLME_GetExtAddr(); // Load the saveExtAddr pointer. //

获得64位的IEEE地址。

}

//The tenth step,执行zdoSendStateChangeMsg()函数

static void zdoSendStateChangeMsg(uint8 state, uint8 taskId) //The tenth

step,

{

osal_event_hdr_t *pMsg = (osal_event_hdr_t *)osal_msg_find(taskId,

ZDO_STATE_CHANGE);

if (NULL == pMsg)

{

if (NULL == (pMsg = (osal_event_hdr_t

*)osal_msg_allocate(sizeof(osal_event_hdr_t))))

{

// Upon failure to notify any EndPoint of the state change,

re-set the ZDO event to

// try again later when more Heap may be available.

osal_set_event(ZDAppTaskID, ZDO_STATE_CHANGE_EVT); //如

果ZDO状态没有任何改变,再一次,跳到

//ZDO_STATE_CHANGE_EVT事件处理

函数。

}

else

{

pMsg->event = ZDO_STATE_CHANGE; //如果ZDO状态改变

了 了,把ZDO_STATE_CHANGE这个消息保存到pMsg

pMsg->status = state;

(void)osal_msg_send(taskId, (uint8 *)pMsg); //转到

MT_TASK.C,去执行The eleven step, MT_ProcessIncomingCommand()函数

}

}

......

}

//The eleventh step,去执行MT_ProcessIncomingCommand()函数

void MT_ProcessIncomingCommand( mtOSALSerialData_t *msg )

{

......

case ZDO_STATE_CHANGE: //The thirteenth step, 接着跳到

MT_ZdoStateChangeCB()函数。

//自此,协调器组网形成(终端设备成功加入网络)

MT_ZdoStateChangeCB((osal_event_hdr_t *)msg);

break;

......

}

//第五步,//初始化玩系统任务事件后,正是开始执行操作系统,此时操作系统

不断的检测有没有任务事件发生,一旦检测到有事件发生,就转 //到相应的处

理函数,进行处理。

void osal_start_system( void ) //第五步,正式执行操作系统

{

#if !defined ( ZBIT ) && !defined ( UBIT )

for(;;) // Forever Loop //死循环

#endif

{

uint8 idx = 0;

osalTimeUpdate();

Hal_ProcessPoll(); // This replaces MT_SerialPoll() and

osal_check_timer().

do {

if (tasksEvents) // Task is highest priority that is ready.

{

break; // 得到待处理的最高优先级任务索引号

idx

}

} while (++idx < tasksCnt);

if (idx < tasksCnt)

{

uint16 events;

halIntState_t intState;

HAL_ENTER_CRITICAL_SECTION(intState); //进入临界区

events = tasksEvents; //提取需要处理的任

务中的事件

tasksEvents = 0; // Clear the Events for this task. //

清除本次任务的事件

HAL_EXIT_CRITICAL_SECTION(intState); //退出临界区

events = (tasksArr)( idx, events ); //通过指针调用任务处理函

数 , 紧接着跳到相应的函数去处理,此为第五步

HAL_ENTER_CRITICAL_SECTION(intState); //进入临界区

tasksEvents |= events; // Add back unprocessed events to the

current task. // 保存未处理的事件

HAL_EXIT_CRITICAL_SECTION(intState); //退出临界区

}

#if defined( POWER_SAVING )

else // Complete pass through all task events with no activity?

{

osal_pwrmgr_powerconserve(); // Put the processor/system

into sleep

}

#endif

}

}

第二个功能:设备间的绑定

/*当我们按下sw2,即JoyStick控杆的右键时,节点发出终端设备绑定请求,

因为我们在SerialApp层,注册过了键盘响应事件,所以,当我们按 下右键

时,我们会在SerialApp_ProcessEvent()函数里找到对应的键盘相应事件*/

UINT16 SerialApp_ProcessEvent( uint8 task_id, UINT16 events ) //当

有事件传递到应用层的时候,执行此处

{

if ( events & SYS_EVENT_MSG ) // 有事件传递过来,故通过这个条件语

句 {

......

case KEY_CHANGE: //键盘触发事件

SerialApp_HandleKeys( ((keyChange_t *)MSGpkt)->state,

((keyChange_t *)MSGpkt)->keys ); //接着跳到相应的按键处理函数去执行

break;

.......

}

}

ZDO终端设备绑定请求:设备能告诉协调器他们想建立绑定表格报告。该协调器

将使协调并在这两个设备上创建绑定表格条目。在这里是以SerialApp例子为

例。

void SerialApp_HandleKeys( uint8 shift, uint8 keys )

{

.......

if ( keys & HAL_KEY_SW_2 ) // Joystick right

{

HalLedSet ( HAL_LED_4, HAL_LED_MODE_OFF ); //终端设

备绑定请求

// Initiate an End Device Bind Request for the mandatory

endpoint de = Addr16Bit;

ddr = 0x0000; // Coordinator 地

址 ZDP_EndDeviceBindReq( &dstAddr,

NLME_GetShortAddr(), //终端设备绑定请

求 SerialApp_

oint, SERIALAPP_PROFID,

SERIALAPP_MAX_CLUSTERS,

(cId_t

*)SerialApp_ClusterList,

SERIALAPP_MAX_CLUSTERS,

(cId_t

*)SerialApp_ClusterList,

FALSE );

}

......

if ( keys & HAL_KEY_SW_4 )

{

HalLedSet ( HAL_LED_4, HAL_LED_MODE_OFF ); //

Initiate a Match Description Request (Service

Discovery) de = AddrBroadcast; //广播地址

ddr =

NWK_BROADCAST_SHORTADDR; ZDP_MatchDescReq( &dstAddr,

NWK_BROADCAST_SHORTADDR, //描述符

匹配请求 这也是两不同匹配方式,使用的按键不

同 SERIALAPP_PROFID,

SERIALAPP_MAX_CLUSTERS,

(cId_t *)SerialApp_ClusterList,

SERIALAPP_MAX_CLUSTERS,

(cId_t *)SerialApp_ClusterList,

FALSE );

}

}

}

说明:从上面可以看到,SW2是发送终端设备绑定请求方式,SW4是发送

描述符匹配请求方式。如果按下SW2的话,使用终端设备绑定请求方式,这里是

要通过终端告诉协调器他们想要建立绑定表格,协调器将协调这两个请求的设

备,在两个设备上建立绑定表格条目。

(1)终端设备向协调器发送终端设备绑定请求

调用ZDP_EndDeviceBindReq()函数发送绑定请求。

ZDP_EndDeviceBindReq( &dstAddr, //目的地址设为

0x0000; NLME_GetShortAddr(),

SerialApp_nt,

//EP号

SERIALAPP_PROFID,//Profile

ID SERIALAPP_MAX_CLUSTERS, //输

入簇的数目

(cId_t *)SerialApp_ClusterList, //输入簇列

表 SERIALAPP_MAX_CLUSTERS, //输出簇

数目

(cId_t *)SerialApp_ClusterList,//输出簇

列表 FALSE );

该函数实际调用无线发送函数将绑定请求发送给协调器节点:默认clusterID

为End_Device_Bind_req,最后通过AF_DataRequest()发送出去.

fillAndSend( &ZDP_TransID, dstAddr, End_Device_Bind_req, len );最后通

过AF_DataRequest()发送出去,这里的&afAddr,是目的地址; &ZDApp_epDesc ,

是端口号; clusterID,是簇号; len+1,是数据的长度;

//ZDP_TmpBuf-1,是数据的内容; transSeq,是数据的顺序号; ZDP_TxOptions,

是发射的一个选项 ; AF_DEFAULT_RADIUS,是一个默认的半径(跳数)。

AF_DataRequest( &afAddr, &ZDApp_epDesc, clusterID,

(uint16)(len+1),

(uint8*)(ZDP_TmpBuf-1), transSeq,

ZDP_TxOptions, AF_DEFAULT_RADIUS );(2) 协调器收到终端设备绑定请求

End_Device_Bind_req

这个信息会传送到ZDO层,在ZDO层的事件处理函数中,调用

ZDApp_ProcessOSALMsg( (osal_event_hdr_t *)msg_ptr );UINT16

ZDApp_event_loop( byte task_id, UINT16 events ){

uint8 *msg_ptr; if ( events & SYS_EVENT_MSG ) {

while ( (msg_ptr = osal_msg_receive( ZDAppTaskID )) ) {

ZDApp_ProcessOSALMsg( (osal_event_hdr_t

*)msg_ptr ); // Release the

memory osal_msg_deallocate( msg_ptr ); }

// Return unprocessed eventsreturn (events ^ SYS_EVENT_MSG);

.....................

}

void ZDApp_ProcessOSALMsg( osal_event_hdr_t *msgPtr ){

// Data Confirmation message fields byte sentEP; //

This should always be 0 byte sentStatus; afDataConfirm_t

*afDataConfirm; switch ( msgPtr->event ) {

// Incoming ZDO Message

case

AF_INCOMING_MSG_CMD: ZDP_IncomingData( (afIncomingMSGPacket_

t *)msgPtr ); break;

................................

}

在ZDP_IncomingData( (afIncomingMSGPacket_t *)msgPtr );函数中void

ZDP_IncomingData( afIncomingMSGPacket_t *pData ){

uint8 x = 0;

uint8 handled; zdoIncomingMsg_t inMsg;

//解析clusterID这个消息 de =

Addr16Bit; ddr =

pData->ddr; adcast =

pData->wasBroadcast; rID =

pData->clusterId; //这个

clusterID,在这里指的是,终端设备发送过来的End_Device_Bind_req这个消

息 tyUse = pData->SecurityUse;

n = pData->ngth-1; =

pData->+1; eq = pData->; handled =

ZDO_SendMsgCBs( &inMsg );

#if defined( MT_ZDO_FUNC )

MT_ZdoRsp( &inMsg );#endif

while ( rID != 0xFFFF ) {

if ( rID == rID ) //在

zdpMsgProcs[]中,查找,看看有没有跟End_Device_Bind_req相匹配的描述

符。 {

( &inMsg ); return;

}

x++;

}

// Handle unhandled messages if

( !handled ) ZDApp_InMsgCB( &inMsg );}

因为ZDO信息处理表zdpMsgProcs[ ]没有对应的End_Device_Bind_req

簇,因此没有调用ZDO信息处理表中的处理函数,但是前面的ZDO_SendMsgCBs()

会把这个终端设备绑定请求发送到登记过这个ZDO信息的任务中去。那这个登记

注册的程序在哪里呢?

对于协调器来说,由于在void ZDApp_Init( byte task_id )函数中调用

了ZDApp_RegisterCBs();面的函数。进行注册了终端绑定请求信息。void

ZDApp_RegisterCBs( void )

{

#if defined ( ZDO_IEEEADDR_REQUEST ) || defined

( REFLECTOR ) ZDO_RegisterForZDOMsg( ZDAppTaskID,

IEEE_addr_rsp );#endif

#if defined ( ZDO_NWKADDR_REQUEST ) || defined

( REFLECTOR ) ZDO_RegisterForZDOMsg( ZDAppTaskID,

NWK_addr_rsp );#endif

#if defined ( ZDO_COORDINATOR )

ZDO_RegisterForZDOMsg( ZDAppTaskID,

Bind_rsp ); ZDO_RegisterForZDOMsg( ZDAppTaskID,

Unbind_rsp ); ZDO_RegisterForZDOMsg( ZDAppTaskID,

End_Device_Bind_req );#endif

#if defined ( REFLECTOR )

ZDO_RegisterForZDOMsg( ZDAppTaskID,

Bind_req ); ZDO_RegisterForZDOMsg( ZDAppTaskID, Unbind_req );#endif

}

因此,协调器节点的 ZDApp 接收到外界输入的数据后,由于注册了 ZDO

反馈消息,即ZDO_CB_MSG,ZDApp 层任务事件处理函数将进行处理:也就是调

用下面的程序。

UINT16 ZDApp_event_loop( byte task_id, UINT16 events ){

uint8 *msg_ptr; if ( events & SYS_EVENT_MSG ) {

while ( (msg_ptr = osal_msg_receive( ZDAppTaskID )) ) {

ZDApp_ProcessOSALMsg( (osal_event_hdr_t

*)msg_ptr ); // Release the

memory osal_msg_deallocate( msg_ptr ); }

// Return unprocessed eventsreturn (events ^ SYS_EVENT_MSG);

..............................

}

在这里调用函数ZDApp_ProcessOSALMsg( (osal_event_hdr_t

*)msg_ptr );在这个函数中我们可以看到对ZDO_CB_MSG事件的处理void

ZDApp_ProcessOSALMsg( osal_event_hdr_t *msgPtr ){

// Data Confirmation message fields byte sentEP; //

This should always be 0 byte sentStatus; afDataConfirm_t

*afDataConfirm; switch ( msgPtr->event ) {

// Incoming ZDO Message case

AF_INCOMING_MSG_CMD: ZDP_IncomingData( (afIncomingMSGPacket_

t *)msgPtr ); break;

case ZDO_CB_MSG: ZDApp_ProcessMsgCBs( (zdoIncomingMsg_t

*)msgPtr ); break;

....................................

}

调用ZDApp_ProcessMsgCBs()函数。在这个函数中根据ClusterID(这

里是 End_Device_Bind_req)选择相对应的匹配描述符处理函数,void

ZDApp_ProcessMsgCBs( zdoIncomingMsg_t *inMsg )

{

.......

case End_Device_Bind_req:

{

ZDEndDeviceBind_t

bindReq; ZDO_ParseEndDeviceBindReq( inMsg,

&bindReq ); //解析绑定请求信

息 ZDO_MatchEndDeviceBind( &bindReq );

//然后向发送绑定请求的节点发送绑定响应消

息: // Freeing the cluster lists - if

allocated. if

( lusters ) osal_mem_free(

sters ); if

( Clusters ) osal_mem_free(

lusters ); }

break;

#endif

}

}

下面是ZDO_MatchEndDeviceBind()函数的源代码

void ZDO_MatchEndDeviceBind( ZDEndDeviceBind_t *bindReq ){

zAddrType_t dstAddr; uint8 sendRsp = FALSE; uint8

status; // Is this the first request? 接收到的是第一个绑定请求 if

( matchED == NULL ) {

// Create match info structure 创建匹配信息结构体 matchED =

(ZDMatchEndDeviceBind_t *)osal_mem_alloc( sizeof

( ZDMatchEndDeviceBind_t ) ); //分配空间 if ( matchED ) {

// Clear the structure 先进行清除操

作 osal_memset( (uint8 *)matchED, 0, sizeof

( ZDMatchEndDeviceBind_t ) ); // Copy the first request's

information 复制第一个请求信息 if

( !ZDO_CopyMatchInfo( &(matchED->ed1), bindReq ) ) //复制不成功

后 {

status = ZDP_NO_ENTRY; sendRsp =

TRUE; }

}

else //分配空间不成功

{

status = ZDP_NO_ENTRY; sendRsp = TRUE; }

if ( !sendRsp ) //分配空间成功 ,复制数据结构成功 {

// Set into the correct state 设置正确的设备状

态 matchED->state = ZDMATCH_WAIT_REQ; // Setup the

timeout 设置计时时间

APS_SetEndDeviceBindTimeout(AIB_MaxBindingTime,

ZDO_EndDeviceBindMatchTimeoutCB ); }

}

else //接收到的不是第一个绑定请求

{

matchED->state = ZDMATCH_SENDING_BINDS; //状态为绑定中 //

Copy the 2nd request's information 拷贝第2个请求信息结构 if

( !ZDO_CopyMatchInfo( &(matchED->ed2), bindReq ) ) //拷贝不成

功 {

status = ZDP_NO_ENTRY; sendRsp =

TRUE; }

// Make a source match for ed1 //对ed1的输出簇ID与ed2

的输入簇ID进行比较,如果有符合的则会返回,相匹配的簇的数目

matchED->ed1numMatched =

ZDO_CompareClusterLists( matchED->

mOutClusters,

matchED->sters, matchED->ed2.n

umInClusters, matchED->ters, ZDOBuildBuf ); if

( matchED->ed1numMatched ) //如果有返回ed1相匹配的

簇 {

// Save the match list 申请空间保存相匹配的簇列

表 matchED->ed1Matched=

osal_mem_alloc( (short)(matchED->ed1numMatched * sizeof

( uint16 )) ); if ( matchED->ed1Matched ) //

分配成功 {

//保存相匹配的簇列表

osal_memcpy(matchED->ed1Matched,ZDOBuildBuf,

(matchED->ed1numMatched * sizeof ( uint16 )) ); }

else //内存空间分配不成功 {

// Allocation error, stop

status = ZDP_NO_ENTRY; sendRsp =

TRUE; }

}

// Make a source match for ed2 以ed2为源 //对ed2的终

端匹配请求和ed1的簇列表相比较,返回相相匹配的簇的数目

matchED->ed2numMatched =

ZDO_CompareClusterLists( matchED->

mOutClusters,

matchED->sters, matchED->ed1.n

umInClusters, matchED->ters, ZDOBuildBuf ); if

( matchED->ed2numMatched ) //如果匹配成功 {

// Save the match list 保存匹配的簇列

表 matchED->ed2Matched =

osal_mem_alloc( (short)(matchED->ed2numMatched * sizeof

( uint16 )) ); if ( matchED->ed2Matched ) {

osal_memcpy( matchED->ed2Matched, ZDOBuildBuf,

(matchED->ed2numMatched * sizeof ( uint16 )) ); }

else

{

// Allocation error, stop status =

ZDP_NO_ENTRY; sendRsp = TRUE; }

}

//如果两个相请求的终端设备,有相匹配的簇,并且保存成功

if ( (sendRsp == FALSE) && (matchED->ed1numMatched ||

matchED->ed2numMatched) ) {

// Do the first unbind/bind state 发送响应信息给两个设

备 ZDMatchSendState( ZDMATCH_REASON_START, ZDP_SUCCESS,

0 ); }

else

{

status = ZDP_NO_MATCH; sendRsp =

TRUE; }

}

if ( sendRsp ) //如果没有相匹配的或匹配不成功 {

// send response to this requester 发送匹配请求响

应 de = Addr16Bit; //设置目的地址是16位的

短地址ddr = bindReq->srcAddr;//发送绑定终端响应函

数status = ZDP_NO_MATCH; ZDP_EndDeviceBindRsp( bindReq->TransSeq,

&dstAddr, status, bindReq->SecurityUse ); if ( matchED->state ==

ZDMATCH_SENDING_BINDS ) {

// send response to first requester de

= Addr16Bit; ddr =

matchED->r; ZDP_EndDeviceBindRsp( matchED->

nsSeq, &dstAddr, status, matchED->tyUse ); }

// Process ended - release memory

used ZDO_RemoveMatchMemory(); }

}

ZDO_MatchEndDeviceBind()函数,如果协调器接收到接收到第一个绑

定请求,则分配内存空间进行保存并计时,如果不是第一个绑定请求,则分别以

第一个和第二个绑定请求为源绑定,进行比较匹配,如果比较匹配成功则发送匹

配成功的信息End_Device_Bind_rsp给两个请求终端。因为在

ZDMatchSendState()函数中也是调用了ZDP_EndDeviceBindRsp()函数,对匹配

请求响应进行了发送。如果匹配不成功则发送匹配失败的信息给两个终端。uint8

ZDMatchSendState( uint8 reason, uint8 status, uint8 TransSeq ){

..............................

else

{

// Send the response messages to requesting devices // send

response to first requester 发送响应信息给第一个请求终

端, ddr =

matchED->r; ZDP_EndDeviceBindRsp( matchED->e

q, &dstAddr, rspStatus, matchED->tyUse );

// send response to second requester 发送响应信息给第二请求终

端 if ( matchED->state == ZDMATCH_SENDING_BINDS ) {

ddr =

matchED->r; ZDP_EndDeviceBindRsp( matchED->

nsSeq, &dstAddr, rspStatus, matchED->tyUse ); }

// Process ended - release memory

used ZDO_RemoveMatchMemory(); }

return ( TRUE );}

(3)终端结点的响应

由于终端节点在 SerialApp.c 中层注册过 End_Device_Bind_rsp 消

息,因此当接收到协调器节点发来的绑定响应消息将交由 SerialApp 任务事件

处理函数处理:UINT16 SerialApp_ProcessEvent( uint8 task_id, UINT16

events ){

if ( events & SYS_EVENT_MSG ) {

afIncomingMSGPacket_t *MSGpkt; while ( (MSGpkt =

(afIncomingMSGPacket_t

*)osal_msg_receive(

SerialApp_TaskID )) ) {

switch ( MSGpkt-> ) {

case ZDO_CB_MSG:

SerialApp_ProcessZDOMsgs( (zdoIncomingMsg_t

*)MSGpkt ); break; ...................................

}

然后,调用 SerialApp_ProcessZDOMsgs()函数。进行事件处理。

static void SerialApp_ProcessZDOMsgs( zdoIncomingMsg_t *inMsg ){

switch ( inMsg->clusterID ) {

case End_Device_Bind_rsp: if ( ZDO_ParseBindRsp( inMsg )

== ZSuccess ) {

// Light LED HalLedSet( HAL_LED_4,

HAL_LED_MODE_ON ); }

#if defined(BLINK_LEDS)

else

{

// Flash LED to show failure HalLedSet

( HAL_LED_4, HAL_LED_MODE_FLASH ); }

#endif

break;

................................

}

第三个功能:实现两个节点间的串口通信

“串口终端1”的数据,如何被“节点 1”所接收,并且发送出去的?

串口数据是由哪层来负责的呢?--HAL。 。 。恩,猜对了。但这

个肯定不是靠猜的,其中的过程就不讲了。 让我们从主循环

(osal_start_system) 的Hal_ProcessPoll函数找下去 (用

source insight

的同学可以用 ctrl +) ,Hal_ProcessPoll ==> HalUARTPoll ==>

HalUARTPollDMA

这个 HalUARTPollDMA 函数里最后有这样一句话:

(HAL_UART_DMA-1, evt); 对 这个函数进行

了调用,ctrl / 搜索这个 ,发现 SerialApp_Init 函数有两

句话:

ckFunc = SerialApp_CallBack;

HalUARTOpen (SERIAL_APP_PORT, &uartConfig);

此处将 这个函数注册成为 SerialApp_CallBack, 也就是说

SerialApp_CallBack函数每次循环中被调用一次,对串口的内容进行查询,如

果 DMA 中接收到了数据,则调用HalUARTRead,将 DMA 数据读至数据 buffer

并通过 AF_DataRequest 函数发送出去,注意:出去的信息的 CLUSTERID(信息

簇ID)号为 SERIALAPP_CLUSTERID1。 总结一下这个过程:

串口数据==>DMA接收==>主循环中通过SerialApp_CallBack 查询==>

从 DMA获取并发送到空中。

具体流程如下:

void SerialApp_Init( uint8 task_id )

{

......

ured =

TRUE; // 2x30 don't care - see uart driver.

te = SERIAL_APP_BAUD;

ntrol = TRUE;

ntrolThreshold = SERIAL_APP_THRESH; // 2x30 don't

care - see uart driver.

Size = SERIAL_APP_RX_SZ; //

2x30 don't care - see uart driver.

Size = SERIAL_APP_TX_SZ; //

2x30 don't care - see uart driver.

meout = SERIAL_APP_IDLE; //

2x30 don't care - see uart driver.

ble =

TRUE; // 2x30 don't care - see uart driver.

ckFunc =

SerialApp_CallBack; //调用SerialApp_CallBack函数,对串口内容

进行查询

HalUARTOpen (SERIAL_APP_PORT, &uartConfig);

......

}

static void SerialApp_CallBack(uint8 port, uint8 event)

{

(void)port;

//如果 DMA 中接收到了数据

if ((event & (HAL_UART_RX_FULL | HAL_UART_RX_ABOUT_FULL |

HAL_UART_RX_TIMEOUT)) &&

#if SERIAL_APP_LOOPBACK

(SerialApp_TxLen < SERIAL_APP_TX_MAX))

#else

!SerialApp_TxLen)

#endif

{

SerialApp_Send(); //调用串口发送函数,将从串口接受到的数据,

发送出去

}

}

static void SerialApp_Send(void)

{

#if SERIAL_APP_LOOPBACK //初始化时,

SERIAL_APP_LOOPBACK=false ,所以不执行if这个预编译,转到else去执行

if (SerialApp_TxLen < SERIAL_APP_TX_MAX)

{

SerialApp_TxLen += HalUARTRead(SERIAL_APP_PORT,

SerialApp_TxBuf+SerialApp_TxLen+1,

SERIAL_APP_TX_MAX-SerialApp_TxLen);

}

if (SerialApp_TxLen)

{

(void)SerialApp_TxAddr;

if (HalUARTWrite(SERIAL_APP_PORT, SerialApp_TxBuf+1,

SerialApp_TxLen))

{

SerialApp_TxLen = 0;

}

else

{

osal_set_event(SerialApp_TaskID, SERIALAPP_SEND_EVT);

}

}

#else

if (!SerialApp_TxLen &&

(SerialApp_TxLen = HalUARTRead(SERIAL_APP_PORT,

SerialApp_TxBuf+1, SERIAL_APP_TX_MAX)))

{

// Pre-pend sequence number to the Tx message.

SerialApp_TxBuf = ++SerialApp_TxSeq;

}

if (SerialApp_TxLen)

{

if (afStatus_SUCCESS !=

AF_DataRequest(&SerialApp_TxAddr, //通过AF_DataRequest()

函数,将数据从空中发送出去

(endPointDesc_t *)&SerialApp_epDesc,

SERIALAPP_CLUSTERID1,

SerialApp_TxLen+1, SerialApp_TxBuf,

&SerialApp_MsgID, 0, AF_DEFAULT_RADIUS))

{

osal_set_event(SerialApp_TaskID,

SERIALAPP_SEND_EVT); //如果数据没有发送成功,重新发送

}

}

#endif

}

UINT16 SerialApp_ProcessEvent( uint8 task_id, UINT16 events )

{

if ( events & SERIALAPP_SEND_EVT ) //当数据没有发送成

功时

{

SerialApp_Send();

return ( events ^ SERIALAPP_SEND_EVT );

}

}

节点2 在收到空中的信号后,如何传递给与其相连的串口终端?

节点 2 从空中捕获到信号后, 在应用层上首先收到信息的就是

SerialApp_ProcessEvent 这个函数了,它收到一个 AF_INCOMING_MSG_CMD 的事

件,并通知 SerialApp_ProcessMSGCmd,执行以下代码 :

UINT16 SerialApp_ProcessEvent( uint8 task_id, UINT16 events ) //当

有事件传递到应用层的时候,执行此处

{

......

while ( (MSGpkt = (afIncomingMSGPacket_t

*)osal_msg_receive( SerialApp_TaskID )) )

{

switch ( MSGpkt-> )

{

......

case AF_INCOMING_MSG_CMD: //在这个实验中,使用

串口通讯时,触发的事件,从空中捕获到信号。

SerialApp_ProcessMSGCmd( MSGpkt ); //处理这个消

break;

......

}

}

}

void SerialApp_ProcessMSGCmd( afIncomingMSGPacket_t *pkt ) //对从空中

捕获到的信号进行处理

{

uint8 stat;

uint8 seqnb;

uint8 delay;

switch ( pkt->clusterId )

{

// A message with a serial data block to be transmitted on the serial

port.

case SERIALAPP_CLUSTERID1: //节点一发送过来的信息的

CLUSTERID(信息簇ID)号为 SERIALAPP_CLUSTERID1

// Store the address for sending and retrying.

osal_memcpy(&SerialApp_RxAddr, &(pkt->srcAddr),

sizeof( afAddrType_t ));

seqnb = pkt->;

// Keep message if not a repeat packet

if ( (seqnb > SerialApp_RxSeq)

|| // Normal

((seqnb < 0x80 ) && ( SerialApp_RxSeq > 0x80)) ) //

Wrap-around

{

// Transmit the data on the serial port.

if ( HalUARTWrite( SERIAL_APP_PORT, pkt->+1,

(pkt->ngth-1) ) ) //通过串口发送数据到PC机

{

// Save for next incoming message

SerialApp_RxSeq = seqnb;

stat = OTA_SUCCESS;

}

else

{

stat = OTA_SER_BUSY;

}

}

else

{

stat = OTA_DUP_MSG;

}

// Select approproiate OTA flow-control delay.

delay = (stat == OTA_SER_BUSY) ? SERIALAPP_NAK_DELAY :

SERIALAPP_ACK_DELAY;

// Build & send OTA response message.

SerialApp_RspBuf = stat;

SerialApp_RspBuf = seqnb;

SerialApp_RspBuf = LO_UINT16( delay );

SerialApp_RspBuf = HI_UINT16( delay );

osal_set_event( SerialApp_TaskID, SERIALAPP_RESP_EVT ); //受

到数据后,向节点1发送一个响应事件,跳到SerialApp_ProcessEvent()

osal_stop_timerEx(SerialApp_TaskID, SERIALAPP_RESP_EVT);

break;

......

}

}

UINT16 SerialApp_ProcessEvent( uint8 task_id, UINT16 events )

{

......

if ( events & SERIALAPP_RESP_EVT ) //串口响应事件,表示成功

接受来自节点1的数据, {

SerialApp_Resp(); //向节点1发送 成功接受的response

return ( events ^ SERIALAPP_RESP_EVT );

}

......

}

static void SerialApp_Resp(void)

{

if (afStatus_SUCCESS !=

AF_DataRequest(&SerialApp_RxAddr, //通过AF_DataRequest

函数,讲接收成功响应从空中发送出去

(endPointDesc_t *)&SerialApp_epDesc,

SERIALAPP_CLUSTERID2,

SERIAL_APP_RSP_CNT, SerialApp_RspBuf,

&SerialApp_MsgID, 0, AF_DEFAULT_RADIUS))

{

osal_set_event(SerialApp_TaskID, SERIALAPP_RESP_EVT); //如

果发送失败,重新发送

}

}

节点1,接收到来自节点2的response。

UINT16 SerialApp_ProcessEvent( uint8 task_id, UINT16 events )

{

......

while ( (MSGpkt = (afIncomingMSGPacket_t

*)osal_msg_receive( SerialApp_TaskID )) )

{

switch ( MSGpkt-> )

{

......

case AF_INCOMING_MSG_CMD: //在这个实验中,使用

串口通讯时,触发的事件,从空中捕获到信号。

SerialApp_ProcessMSGCmd( MSGpkt ); //处理这个消

break;

......

}

}

}

SERIALAPP_CLUSTERID2代表接收到发送成功的response,取消自动重发,如果

不,自动重发。

void SerialApp_ProcessMSGCmd( afIncomingMSGPacket_t *pkt )

{

......

// A response to a received serial data block.

case SERIALAPP_CLUSTERID2: //SerialWsn_CLUSTERID2代

表接收到发送成功的response

if ((pkt-> == SerialApp_TxSeq) &&

((pkt-> == OTA_SUCCESS) || (pkt-> ==

OTA_DUP_MSG)))

{

SerialApp_TxLen = 0;

osal_stop_timerEx(SerialApp_TaskID,

SERIALAPP_SEND_EVT); //当收到发送成功的response,停止自动从发

}

else

{

// Re-start timeout according to delay sent from other device.

delay = BUILD_UINT16( pkt->, pkt-> );

osal_start_timerEx( SerialApp_TaskID, SERIALAPP_SEND_EVT,

delay ); //没有收到成功的response,自动重发

}

break;

default:

break;

}

附录:源程序

.h文件如下:

#ifndef SERIALAPP_H

#define SERIALAPP_H

#ifdef __cplusplus

extern "C"

{

#endif

#include "ZComDef.h"

// These constants are only for example and should be changed to the

// device's needs

#define SERIALAPP_ENDPOINT 11

#define SERIALAPP_PROFID 0x0F05

#define SERIALAPP_DEVICEID 0x0001

#define SERIALAPP_DEVICE_VERSION 0

#define SERIALAPP_FLAGS 0

#define SERIALAPP_MAX_CLUSTERS 2

#define SERIALAPP_CLUSTERID1 1

#define SERIALAPP_CLUSTERID2 2

#define SERIALAPP_SEND_EVT 0x0001

#define SERIALAPP_RESP_EVT 0x0002

// OTA Flow Control Delays

#define SERIALAPP_ACK_DELAY 1

#define SERIALAPP_NAK_DELAY 16

// OTA Flow Control Status

#define OTA_SUCCESS ZSuccess

#define OTA_DUP_MSG (ZSuccess+1)

#define OTA_SER_BUSY (ZSuccess+2)

extern byte SerialApp_TaskID;

extern UINT16 SerialApp_ProcessEvent( byte task_id, UINT16 events );

#ifdef __cplusplus

}

#endif

#endif /* SERIALAPP_H */

.c文件如下

#include "AF.h"

#include "OnBoard.h"

#include "OSAL_Tasks.h"

#include "SerialApp.h"

#include "ZDApp.h"

#include "ZDObject.h"

#include "ZDProfile.h"

#include "hal_drivers.h"

#include "hal_key.h"

#if defined ( LCD_SUPPORTED )

#include "hal_lcd.h"

#endif

#include "hal_led.h"

#include "hal_uart.h"

#if !defined( SERIAL_APP_PORT )

#define SERIAL_APP_PORT 0

#endif

#if !defined( SERIAL_APP_BAUD )

#define SERIAL_APP_BAUD HAL_UART_BR_38400

//#define SERIAL_APP_BAUD HAL_UART_BR_115200

#endif

// When the Rx buf space is less than this threshold, invoke the Rx

callback.

#if !defined( SERIAL_APP_THRESH )

#define SERIAL_APP_THRESH 64

#endif

#if !defined( SERIAL_APP_RX_SZ )

#define SERIAL_APP_RX_SZ 128

#endif

#if !defined( SERIAL_APP_TX_SZ )

#define SERIAL_APP_TX_SZ 128

#endif

// Millisecs of idle time after a byte is received before invoking Rx

callback.

#if !defined( SERIAL_APP_IDLE )

#define SERIAL_APP_IDLE 6

#endif

// Loopback Rx bytes to Tx for throughput testing.

#if !defined( SERIAL_APP_LOOPBACK )

#define SERIAL_APP_LOOPBACK FALSE

#endif

// This is the max byte count per OTA message.

#if !defined( SERIAL_APP_TX_MAX )

#define SERIAL_APP_TX_MAX 80

#endif

#define SERIAL_APP_RSP_CNT 4

// This list should be filled with Application specific Cluster IDs.

const cId_t SerialApp_ClusterList[SERIALAPP_MAX_CLUSTERS] =

{

SERIALAPP_CLUSTERID1,

SERIALAPP_CLUSTERID2

};

const SimpleDescriptionFormat_t SerialApp_SimpleDesc =

{

SERIALAPP_ENDPOINT, // int Endpoint;

SERIALAPP_PROFID, // uint16 AppProfId[2];

SERIALAPP_DEVICEID, // uint16 AppDeviceId[2];

SERIALAPP_DEVICE_VERSION, // int AppDevVer:4;

SERIALAPP_FLAGS, // int AppFlags:4;

SERIALAPP_MAX_CLUSTERS, // byte AppNumInClusters;

(cId_t *)SerialApp_ClusterList, // byte *pAppInClusterList;

SERIALAPP_MAX_CLUSTERS, // byte AppNumOutClusters;

(cId_t *)SerialApp_ClusterList // byte *pAppOutClusterList;

};

const endPointDesc_t SerialApp_epDesc =

{

SERIALAPP_ENDPOINT,

&SerialApp_TaskID,

(SimpleDescriptionFormat_t *)&SerialApp_SimpleDesc,

noLatencyReqs

};

uint8 SerialApp_TaskID; // Task ID for internal task/event

processing.

static uint8 SerialApp_MsgID;

static afAddrType_t SerialApp_TxAddr;

static uint8 SerialApp_TxSeq;

static uint8 SerialApp_TxBuf[SERIAL_APP_TX_MAX+1];

static uint8 SerialApp_TxLen;

static afAddrType_t SerialApp_RxAddr;

static uint8 SerialApp_RxSeq;

static uint8 SerialApp_RspBuf[SERIAL_APP_RSP_CNT];

static void SerialApp_ProcessZDOMsgs( zdoIncomingMsg_t *inMsg );

static void SerialApp_HandleKeys( uint8 shift, uint8 keys );

static void SerialApp_ProcessMSGCmd( afIncomingMSGPacket_t *pkt );

static void SerialApp_Send(void);

static void SerialApp_Resp(void);

static void SerialApp_CallBack(uint8 port, uint8 event);

void SerialApp_Init( uint8 task_id )

{

halUARTCfg_t uartConfig;

SerialApp_TaskID = task_id;

SerialApp_RxSeq = 0xC3;

afRegister( (endPointDesc_t *)&SerialApp_epDesc );

RegisterForKeys( task_id );

ured = TRUE; // 2x30 don't

care - see uart driver.

te = SERIAL_APP_BAUD;

ntrol = TRUE;

ntrolThreshold = SERIAL_APP_THRESH; // 2x30 don't

care - see uart driver.

Size = SERIAL_APP_RX_SZ; // 2x30 don't

care - see uart driver.

Size = SERIAL_APP_TX_SZ; // 2x30 don't

care - see uart driver.

meout = SERIAL_APP_IDLE; // 2x30 don't

care - see uart driver.

ble = TRUE; // 2x30 don't

care - see uart driver.

ckFunc = SerialApp_CallBack;

HalUARTOpen (SERIAL_APP_PORT, &uartConfig);

#if defined ( LCD_SUPPORTED )

HalLcdWriteString( "SerialApp", HAL_LCD_LINE_2 );

#endif

ZDO_RegisterForZDOMsg( SerialApp_TaskID, End_Device_Bind_rsp );

ZDO_RegisterForZDOMsg( SerialApp_TaskID, Match_Desc_rsp );

}

UINT16 SerialApp_ProcessEvent( uint8 task_id, UINT16 events )

{

(void)task_id; // Intentionally unreferenced parameter

if ( events & SYS_EVENT_MSG )

{

afIncomingMSGPacket_t *MSGpkt;

while ( (MSGpkt = (afIncomingMSGPacket_t

*)osal_msg_receive( SerialApp_TaskID )) )

{

switch ( MSGpkt-> )

{

case ZDO_CB_MSG:

SerialApp_ProcessZDOMsgs( (zdoIncomingMsg_t *)MSGpkt );

break;

case KEY_CHANGE:

SerialApp_HandleKeys( ((keyChange_t *)MSGpkt)->state,

((keyChange_t *)MSGpkt)->keys );

break;

case AF_INCOMING_MSG_CMD:

SerialApp_ProcessMSGCmd( MSGpkt );

break;

default:

break;

}

osal_msg_deallocate( (uint8 *)MSGpkt );

}

return ( events ^ SYS_EVENT_MSG );

}

if ( events & SERIALAPP_SEND_EVT )

{

SerialApp_Send();

return ( events ^ SERIALAPP_SEND_EVT );

}

if ( events & SERIALAPP_RESP_EVT )

{

SerialApp_Resp();

return ( events ^ SERIALAPP_RESP_EVT );

}

return ( 0 ); // Discard unknown events.

}

static void SerialApp_ProcessZDOMsgs( zdoIncomingMsg_t *inMsg )

{

switch ( inMsg->clusterID )

{

case End_Device_Bind_rsp:

if ( ZDO_ParseBindRsp( inMsg ) == ZSuccess )

{

// Light LED

HalLedSet( HAL_LED_4, HAL_LED_MODE_ON );

}

#if defined(BLINK_LEDS)

else

{

// Flash LED to show failure

HalLedSet ( HAL_LED_4, HAL_LED_MODE_FLASH );

}

#endif

break;

case Match_Desc_rsp:

{

ZDO_ActiveEndpointRsp_t *pRsp = ZDO_ParseEPListRsp( inMsg );

if ( pRsp )

{

if ( pRsp->status == ZSuccess && pRsp->cnt )

{

SerialApp_de = (afAddrMode_t)Addr16Bit;

SerialApp_ddr = pRsp->nwkAddr;

// Take the first endpoint, Can be changed to search through

endpoints

SerialApp_nt = pRsp->epList[0];

// Light LED

HalLedSet( HAL_LED_4, HAL_LED_MODE_ON );

}

osal_mem_free( pRsp );

}

}

break;

}

}

void SerialApp_HandleKeys( uint8 shift, uint8 keys )

{

zAddrType_t txAddr;

if ( shift )

{

if ( keys & HAL_KEY_SW_1 )

{

}

if ( keys & HAL_KEY_SW_2 )

{

}

if ( keys & HAL_KEY_SW_3 )

{

}

if ( keys & HAL_KEY_SW_4 )

{

}

}

else

{

if ( keys & HAL_KEY_SW_1 )

{

}

if ( keys & HAL_KEY_SW_2 )

{

HalLedSet ( HAL_LED_4, HAL_LED_MODE_OFF );

// Initiate an End Device Bind Request for the mandatory endpoint

de = Addr16Bit;

ddr = 0x0000; // Coordinator

ZDP_EndDeviceBindReq( &txAddr, NLME_GetShortAddr(),

SerialApp_nt,

SERIALAPP_PROFID,

SERIALAPP_MAX_CLUSTERS, (cId_t

*)SerialApp_ClusterList,

SERIALAPP_MAX_CLUSTERS, (cId_t

*)SerialApp_ClusterList,

FALSE );

}

if ( keys & HAL_KEY_SW_3 )

{

}

if ( keys & HAL_KEY_SW_4 )

{

HalLedSet ( HAL_LED_4, HAL_LED_MODE_OFF );

// Initiate a Match Description Request (Service Discovery)

de = AddrBroadcast;

ddr = NWK_BROADCAST_SHORTADDR;

ZDP_MatchDescReq( &txAddr, NWK_BROADCAST_SHORTADDR,

SERIALAPP_PROFID,

SERIALAPP_MAX_CLUSTERS, (cId_t

*)SerialApp_ClusterList,

SERIALAPP_MAX_CLUSTERS, (cId_t

*)SerialApp_ClusterList,

FALSE );

}

}

}

void SerialApp_ProcessMSGCmd( afIncomingMSGPacket_t *pkt )

{

uint8 stat;

uint8 seqnb;

uint8 delay;

switch ( pkt->clusterId )

{

// A message with a serial data block to be transmitted on the serial

port.

case SERIALAPP_CLUSTERID1:

// Store the address for sending and retrying.

osal_memcpy(&SerialApp_RxAddr, &(pkt->srcAddr),

sizeof( afAddrType_t ));

seqnb = pkt->[0];

// Keep message if not a repeat packet

if ( (seqnb > SerialApp_RxSeq) || // Normal

((seqnb < 0x80 ) && ( SerialApp_RxSeq > 0x80)) ) // Wrap-around

{

// Transmit the data on the serial port.

if ( HalUARTWrite( SERIAL_APP_PORT, pkt->+1,

(pkt->ngth-1) ) )

{

// Save for next incoming message

SerialApp_RxSeq = seqnb;

stat = OTA_SUCCESS;

}

else

{

stat = OTA_SER_BUSY;

}

}

else

{

stat = OTA_DUP_MSG;

}

// Select approproiate OTA flow-control delay.

delay = (stat == OTA_SER_BUSY) ? SERIALAPP_NAK_DELAY :

SERIALAPP_ACK_DELAY;

// Build & send OTA response message.

SerialApp_RspBuf[0] = stat;

SerialApp_RspBuf[1] = seqnb;

SerialApp_RspBuf[2] = LO_UINT16( delay );

SerialApp_RspBuf[3] = HI_UINT16( delay );

osal_set_event( SerialApp_TaskID, SERIALAPP_RESP_EVT );

osal_stop_timerEx(SerialApp_TaskID, SERIALAPP_RESP_EVT);

break;

// A response to a received serial data block.

case SERIALAPP_CLUSTERID2:

if ((pkt->[1] == SerialApp_TxSeq) &&

((pkt->[0] == OTA_SUCCESS) || (pkt->[0] ==

OTA_DUP_MSG)))

{

SerialApp_TxLen = 0;

osal_stop_timerEx(SerialApp_TaskID, SERIALAPP_SEND_EVT);

}

else

{

// Re-start timeout according to delay sent from other device.

delay = BUILD_UINT16( pkt->[2], pkt->[3] );

osal_start_timerEx( SerialApp_TaskID, SERIALAPP_SEND_EVT,

delay );

}

break;

default:

break;

}

}

static void SerialApp_Send(void)

{

#if SERIAL_APP_LOOPBACK

if (SerialApp_TxLen < SERIAL_APP_TX_MAX)

{

SerialApp_TxLen += HalUARTRead(SERIAL_APP_PORT,

SerialApp_TxBuf+SerialApp_TxLen+1,

SERIAL_APP_TX_MAX-SerialApp_TxLen);

}

if (SerialApp_TxLen)

{

(void)SerialApp_TxAddr;

if (HalUARTWrite(SERIAL_APP_PORT, SerialApp_TxBuf+1,

SerialApp_TxLen))

{

SerialApp_TxLen = 0;

}

else

{

osal_set_event(SerialApp_TaskID, SERIALAPP_SEND_EVT);

}

}

#else

if (!SerialApp_TxLen &&

(SerialApp_TxLen = HalUARTRead(SERIAL_APP_PORT,

SerialApp_TxBuf+1, SERIAL_APP_TX_MAX)))

{

// Pre-pend sequence number to the Tx message.

SerialApp_TxBuf[0] = ++SerialApp_TxSeq;

}

if (SerialApp_TxLen)

{

if (afStatus_SUCCESS != AF_DataRequest(&SerialApp_TxAddr,

(endPointDesc_t

*)&SerialApp_epDesc,

SERIALAPP_CLUSTERID1,

SerialApp_TxLen+1,

SerialApp_TxBuf,

&SerialApp_MsgID, 0,

AF_DEFAULT_RADIUS))

{

osal_set_event(SerialApp_TaskID, SERIALAPP_SEND_EVT);

}


本文标签: 发送 函数 网络 请求