您的位置:首页 > 理论基础 > 计算机网络

1-12 实验10 网络管理实验2 已知网络地址查询MAC地址

2013-10-18 21:19 495 查看
已知网络地址查询MAC地址

1、实验内容: 协调器上电后建立网路,路由器自动加入网络。然后路由器调用调用相关的API函数获得某一网络号节点的MAC地址,然后通过串口将其发送到PC端的串口调试助手

2、知识补充:获得某一网络号节点的MAC地址的API函数。

ZDP_IEEEAddrReq(uint16 shortAddr,byte ReqType,byte StartIndex,byte SecurtyEnable);除了shortAddr这个参数外,其他均填0即可

3、程序设计

协调器程序设计(同1-11 实验9 网络管理实验1 获取自身的和父节点网络地址、MAC地址 的Coordinator.c。http://blog.csdn.net/gdliweibing/article/details/12837721)
路由器程序设计(在同1-11 实验9 网络管理实验1 获取自身的和父节点网络地址、MAC地址
的Enddevice.c的基础上添加、修改代码)。如下:

//Enddevice.c
#include "OSAL.h"
#include "AF.h"
#include "ZDApp.h"
#include "ZDObject.h"
#include "ZDProfile.h"
#include <string.h>
#include "Coordinator.h"
#include "DebugTrace.h"

#if !defined(WIN32)
#include "OnBoard.h"
#endif

#include "hal_lcd.h"
#include "hal_led.h"
#include "hal_key.h"
#include "hal_uart.h"

#define SHOW_INFO_EVENT 0x01

const cId_t GenericApp_ClusterList[GENERICAPP_MAX_CLUSTERS]=
{
GENERICAPP_CLUSTERID
};

//初始化端口描述符
const SimpleDescriptionFormat_t GenericApp_SimpleDesc=
{
GENERICAPP_ENDPOINT,
GENERICAPP_PROFID,
GENERICAPP_DEVICEID,
GENERICAPP_DEVICE_VERSION,
GENERICAPP_FLAGS,
0,
(cId_t*)NULL,
GENERICAPP_MAX_CLUSTERS,
(cId_t*)GenericApp_ClusterList
};

endPointDesc_t GenericApp_epDesc;//节点描述符
byte GenericApp_TaskID;          //任务优先级
byte GenericApp_TransID;         //数据发送序列号
devStates_t GenericApp_NwkState;//保存节点状态

void ShowInfo(void);
void To_string(uint8* dest,char* src,uint8 length);

void GenericApp_ProcessZDOMsgs(zdoIncomingMsg_t *inMsg);
//增加这个函数的目的是对ZDO_CB_MSG消息响应

typedef struct RFTXBUF
{
uint8 myNWK[4];
uint8 myMAC[16];
uint8 pNWK[4];
uint8 pMAC[16];
}RFTX;

//任务初始化函数
void GenericApp_Init(byte task_id)
{
GenericApp_TaskID     = task_id;//初始化任务优先级
GenericApp_NwkState   =DEV_INIT; //初始化为DEV_INIT,表节点没有连接到ZigBee网络
GenericApp_TransID    =0;        //发送数据包的序列号初始化为0
//对节点描述符进行初始化
GenericApp_epDesc.endPoint=GENERICAPP_ENDPOINT;
GenericApp_epDesc.task_id =&GenericApp_TaskID;
GenericApp_epDesc.simpleDesc=(SimpleDescriptionFormat_t*)&GenericApp_SimpleDesc;
GenericApp_epDesc.latencyReq=noLatencyReqs;
//afRegister()函数将节点描述符进行注册,注册后才可以使用OSAL提供的系统服务
afRegister(&GenericApp_epDesc);
halUARTCfg_t uartConfig;
uartConfig.configured=TRUE;
uartConfig.baudRate  =HAL_UART_BR_115200;
uartConfig.flowControl=FALSE;
uartConfig.callBackFunc=NULL;
HalUARTOpen(0,&uartConfig);//这一句至关重要,刚才漏写啦!!!!!!!!!!!!!!!

ZDO_RegisterForZDOMsg(GenericApp_TaskID,IEEE_addr_rsp);//对IEEE_addr_rsp消息响应的注册
}

//消息处理函数
UINT16 GenericApp_ProcessEvent(byte task_id,UINT16 events)
{
afIncomingMSGPacket_t* MSGpkt;
if(events&SYS_EVENT_MSG)
{
MSGpkt=(afIncomingMSGPacket_t*)osal_msg_receive(GenericApp_TaskID);
while(MSGpkt)
{
switch(MSGpkt->hdr.event)
{
case ZDO_CB_MSG:
GenericApp_ProcessZDOMsgs((zdoIncomingMsg_t*)MSGpkt);//注意这个
break;
case ZDO_STATE_CHANGE:   //加入网络后,加入族中
GenericApp_NwkState=(devStates_t)(MSGpkt->hdr.status);//读取节点的设备类型
if(GenericApp_NwkState==DEV_ROUTER)
{
HalLedBlink(HAL_LED_1,0,50,500);    //已加入LED1 闪烁
osal_set_event(GenericApp_TaskID,SHOW_INFO_EVENT);
}
break;
default:
break;
}
osal_msg_deallocate((uint8*)MSGpkt);
MSGpkt=(afIncomingMSGPacket_t*)osal_msg_receive(GenericApp_TaskID);
}
return (events^SYS_EVENT_MSG);
}

if(events&SHOW_INFO_EVENT)
{
HalLedBlink(HAL_LED_2,0,50,500);    //结果一发送到串口LED2 闪烁
ShowInfo();

ZDP_IEEEAddrReq(0x0000,0,0,0);//请求协调器IEEE地址

osal_start_timerEx(GenericApp_TaskID,SHOW_INFO_EVENT,5000);
return (events^SHOW_INFO_EVENT);//清除事件标志
}
return 0;
}

void GenericApp_ProcessZDOMsgs(zdoIncomingMsg_t* inMsg)
{
char buf[16];
char changeline[2]={0x0A,0x0D};
switch(inMsg->clusterID)
{
case IEEE_addr_rsp:
{
ZDO_NwkIEEEAddrResp_t* pRsp=ZDO_ParseAddrRsp(inMsg);//对接受到的数据包进行解析,解析完后pRsp指向数据包的存放地址
if(pRsp)
{
if(pRsp->status==ZSuccess)//解析是否正确
{
To_string(buf,pRsp->extAddr,8);//把MAC地址 转换为16进制形式存放
HalUARTWrite(0,"Coordinator MAC:",osal_strlen("Coordinator MAC:"));
HalUARTWrite(0,buf,16);
HalUARTWrite(0,changeline,2);
}
osal_mem_free(pRsp);//调用该函数释放数据包缓冲区
}
}
break;
}
}

void ShowInfo(void)
{
RFTX rftx;
uint8 buf[8];//内存分配不足时,不比的PC,单片机是不会提示的,编译能通过,能下载,但运行时 会出错,但你看不到错误,只有你查看代码时才可能发现错误。
uint8 changline[2]={0x0A,0x0D};
uint16 nwk;
nwk=NLME_GetShortAddr();
To_string(rftx.myNWK,(uint8*)&nwk,2);      //uint8*4 uint16=uint8*2
To_string(rftx.myMAC,NLME_GetExtAddr(),8);//uint8*16 byte*8

nwk=NLME_GetCoordShortAddr();
To_string(rftx.pNWK,(uint8*)&nwk,2);
NLME_GetCoordExtAddr(buf);
To_string(rftx.pMAC,buf,8);
HalUARTWrite(0,"NWK:",osal_strlen("NWK:"));
HalUARTWrite(0,rftx.myNWK,4);
HalUARTWrite(0,"  MAC:",osal_strlen("  MAC:"));
HalUARTWrite(0,rftx.myMAC,16);
HalUARTWrite(0,"  p-NWK:",osal_strlen("  p-NWK:"));
HalUARTWrite(0,rftx.pNWK,4);
HalUARTWrite(0,"  p-MAC:",osal_strlen("  p-MAC:"));
HalUARTWrite(0,rftx.pMAC,16);
HalUARTWrite(0,changline,2);
}

void To_string(uint8 *dest,char* src,uint8 length)//二进制书转化为十六进制数
{
uint8* xad;
uint8 i=0;
uint8 ch;
xad=src+length-1;
for(i=0;i<length;i++,xad--)
{
ch=(*xad>>4)&0x0F;  //除以十六
dest[i<<1]=ch+((ch<10)?'0':'7');
ch=*xad&0x0F;
dest[(i<<1)+1]=ch+((ch<10)?'0':'7');
}
}

添加代码有:

1、调用ZDP_IEEEAddrRsp()函数发送地址请求。

ZDP_IEEEAddrReq(0x0000,0,0,0);//请求协调器IEEE地址

2、等待协调器发送自身的IEEE地址(协议栈自动完成,无需添加代码)

3、添加ZDO_CB_MSG消息相应函数,并调用ZDO_ParseAddrRsp()函数对数据包进行解析得到所需的IEEE地址。

ZDO_RegisterForZDOMsg(GenericApp_TaskID,IEEE_addr_rsp);//对IEEE_addr_rsp消息响应的注册(这个关ZDO_CB_MSG消息的事????)

添加ZDO_CB_MSG消息相应函数

while(MSGpkt)
{
switch(MSGpkt->hdr.event)
{
case ZDO_CB_MSG:
GenericApp_ProcessZDOMsgs((zdoIncomingMsg_t*)MSGpkt);//注意这个
break;

调用ZDO_ParseAddrRsp()函数对数据包进行解析得到所需的IEEE地址

void GenericApp_ProcessZDOMsgs(zdoIncomingMsg_t* inMsg)
{
char buf[16];
char changeline[2]={0x0A,0x0D};
switch(inMsg->clusterID)
{
case IEEE_addr_rsp:
{
ZDO_NwkIEEEAddrResp_t* pRsp=ZDO_ParseAddrRsp(inMsg);//对接受到的数据包进行解析,解析完后pRsp指向数据包的存放地址
if(pRsp)
{
if(pRsp->status==ZSuccess)//解析是否正确
{
To_string(buf,pRsp->extAddr,8);//把MAC地址 转换为16进制形式存放
HalUARTWrite(0,"Coordinator MAC:",osal_strlen("Coordinator MAC:"));
HalUARTWrite(0,buf,16);
HalUARTWrite(0,changeline,2);
}
osal_mem_free(pRsp);//调用该函数释放数据包缓冲区
}
}
break;
}
}


上面的

ZDO_NwkIEEEAddrResp_t* pRsp=ZDO_ParseAddrRsp(inMsg);//对接受到的数据包进行解析,解析完后pRsp指向数据包的存放地址


该函数的返回值类型为ZDO_NwkIEEEAddrResp_t,该结构体如下:

// Filename: ZDObject.h

typedef struct

{

uint8 status;

uint16 nwkAddr;

uint8 extAddr[Z_EXTADDR_LEN];

uint8 numAssocDevs;

uint8 startIndex;

uint16 devList[];

} ZDO_NwkIEEEAddrResp_t;

4、实验结果

内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: