?? arp.c
字號:
#define ARP_GLOBALS
#include "..\User\includes.h"
//#include "../include/cfg_net.h"
typedef struct { /* arp報文的內容總長28字節*/
uint16 HardwareType; /*以太網為0x0001*/
uint16 ProtocalType; /*ip 為0X0800*/
uint8 HardwareLen; /*=0X06*/
uint8 ProtocalLen; /*=0X04*/
uint16 Operation; /*操作 0X0001為請求 0X0002為應答 */
/*0X0003為反向地址請求 0X0004為反向地址應答*/
uint8 SourceMacId[6]; /*源網卡地址*/
uint8 SourceIp[4]; /*源IP地址*/
uint8 DestMacId[6]; /*目的網卡地址*/
uint8 DestId[4]; /*目的IP地址*/
} arp ;
uint8 ARP_REP_MAC[6]={0XFF,0XFF,0XFF,0XFF,0XFF,0XFF};
uint8 Ping_Ip_Address[4];
uint8 Ping_IP_TTL;
//uint8 My_Ip_Address[4]=MY_IP;
//uint8 My_Gateway_Ip_Address[4]=MY_GATEWAY;
//uint8 MY_IP_MARK[4]=IP_MARK_SETTING;
//struct a_t ARP_TERM[MAX_ARP_TERM];
//===========================================================
/**********************************************************************
**函數原型: unsigned char Arp_Answer(unsigned char EX_RAM * ARP_REC_PTR)
**入口參數: * ARP_REC_PTR :接收到的ARP請求包的指針
**出口參數: unsigned char
**返 回 值: 0 :本子網內的ARP緩沖區已經有MAC
** 1 :目標IP不是本機不作回應
** 2 :本子網內的,但ARP緩沖區還沒有MAC
** 3 :不是本子網的ARP請求
** 4 :ARP緩沖區已經滿
**說 明: 對ARP請求報文的應答:填充本地物理地址,將ARP操作改為回答.
************************************************************************/
uint8 Arp_Answer(uint8 * ARP_REC_PTR,uint8 num)
{
struct _pkst TxdArp;
uint8 SEND_ARP_MAC[6];
uint8 i;
//如果目標IP地址是本機IP
if (((arp*)ARP_REC_PTR)->DestId[0]==NetPort[num].My_Ip[0])
if (((arp*)ARP_REC_PTR)->DestId[1]==NetPort[num].My_Ip[1])
if (((arp*)ARP_REC_PTR)->DestId[2]==NetPort[num].My_Ip[2])
if (((arp*)ARP_REC_PTR)->DestId[3]==NetPort[num].My_Ip[3])
{ //表示是要解析本地IP的請求
for(i=0;i<4;i++) //復制對方IP地址,填充源地址
{
SEND_ARP_MAC[i]=((arp*)ARP_REC_PTR)->SourceIp[i];
((arp*)ARP_REC_PTR)->SourceIp[i]=NetPort[num].My_Ip[i];
((arp*)ARP_REC_PTR)->DestId[i]=SEND_ARP_MAC[i];
}
for(i=0;i<6;i++)
{ //復制對方物理地址或網關地址
SEND_ARP_MAC[i]=((arp*)ARP_REC_PTR)->SourceMacId[i];
((arp*)ARP_REC_PTR)->SourceMacId[i]=NetPort[num].My_Mac[i];
((arp*)ARP_REC_PTR)->DestMacId[i]=SEND_ARP_MAC[i];
}
#ifdef Big_End
((arp*)ARP_REC_PTR)->Operation=0x0002; //表明數據幀為ARP應答
#endif
#ifdef Little_End
((arp*)ARP_REC_PTR)->Operation=0x0200; //表明數據幀為ARP應答
#endif
TxdArp.STPTR=NULL;
TxdArp.length=0x60;
TxdArp.DAPTR=ARP_REC_PTR;
Send_ethernet_Frame(&TxdArp,SEND_ARP_MAC,ARP_PACKED,num);//發送ARP應答幀
//如果發送方屬于本網段
if((((arp*)ARP_REC_PTR)->SourceIp[0]&NetPort[num].My_Ip_Mark[0])==(NetPort[num].My_Ip[0]&NetPort[num].My_Ip_Mark[0]))
if((((arp*)ARP_REC_PTR)->SourceIp[1]&NetPort[num].My_Ip_Mark[1])==(NetPort[num].My_Ip[1]&NetPort[num].My_Ip_Mark[1]))
if((((arp*)ARP_REC_PTR)->SourceIp[2]&NetPort[num].My_Ip_Mark[2])==(NetPort[num].My_Ip[2]&NetPort[num].My_Ip_Mark[2]))
if((((arp*)ARP_REC_PTR)->SourceIp[3]&NetPort[num].My_Ip_Mark[3])==(NetPort[num].My_Ip[3]&NetPort[num].My_Ip_Mark[3]))
{
//查找有否屬于該IP的對應MAC表
for(i=0;i<MAX_ARP_TERM;i++)
{
if(NetPort[num].ARP_TERM[i].IP_NUM[2]==((arp*)ARP_REC_PTR)->SourceIp[2])
if(NetPort[num].ARP_TERM[i].IP_NUM[3]==((arp*)ARP_REC_PTR)->SourceIp[3])
if(NetPort[num].ARP_TERM[i].TTL>0)
{//有則刷新
NetPort[num].ARP_TERM[i].TTL=100;
return(0);
}
}
//查找有否空的MAC表項
for(i=0;i<MAX_ARP_TERM;i++)
{
if(NetPort[num].ARP_TERM[i].TTL==0)
{//有則保存
NetPort[num].ARP_TERM[i].IP_NUM[0]=((arp*)ARP_REC_PTR)->SourceIp[0];
NetPort[num].ARP_TERM[i].IP_NUM[1]=((arp*)ARP_REC_PTR)->SourceIp[1];
NetPort[num].ARP_TERM[i].IP_NUM[2]=((arp*)ARP_REC_PTR)->SourceIp[2];
NetPort[num].ARP_TERM[i].IP_NUM[3]=((arp*)ARP_REC_PTR)->SourceIp[3];
NetPort[num].ARP_TERM[i].MAC_NUM[0]=((arp*)ARP_REC_PTR)->SourceMacId[0];
NetPort[num].ARP_TERM[i].MAC_NUM[1]=((arp*)ARP_REC_PTR)->SourceMacId[1];
NetPort[num].ARP_TERM[i].MAC_NUM[2]=((arp*)ARP_REC_PTR)->SourceMacId[2];
NetPort[num].ARP_TERM[i].MAC_NUM[3]=((arp*)ARP_REC_PTR)->SourceMacId[3];
NetPort[num].ARP_TERM[i].MAC_NUM[4]=((arp*)ARP_REC_PTR)->SourceMacId[4];
NetPort[num].ARP_TERM[i].MAC_NUM[5]=((arp*)ARP_REC_PTR)->SourceMacId[5];
NetPort[num].ARP_TERM[i].TTL=100;
return(2);
}
}//FOR
//MAC表已經滿
return(4);
}//IF ARP
//不屬于同一網段的
return (3);
}
//目標IP不是本機
return (1);
}
uint8 REC_ARP_REQ(uint8 * ARP_REC_REQ_PTR,uint8 num)
{
uint8 i;
//======================================================================
if(((arp*)ARP_REC_REQ_PTR)->SourceIp[0]==NetPort[num].My_Gateway[0])
if(((arp*)ARP_REC_REQ_PTR)->SourceIp[1]==NetPort[num].My_Gateway[1])
if(((arp*)ARP_REC_REQ_PTR)->SourceIp[2]==NetPort[num].My_Gateway[2])
if(((arp*)ARP_REC_REQ_PTR)->SourceIp[3]==NetPort[num].My_Gateway[3])
{ //表示是網關對ARP請求的回答.
for (i=0;i<6;i++)
{
NetPort[num].My_Gateway_Mac[i]=((arp*)ARP_REC_REQ_PTR)->SourceMacId[i];
}
NetPort[num].Gateway_IP_TTL=100; //表示網關地址已得到解析
//return(3);
}
//如果發送方屬于本網段
if((((arp*)ARP_REC_REQ_PTR)->SourceIp[0]&NetPort[num].My_Ip_Mark[0])==(NetPort[num].My_Ip[0]&NetPort[num].My_Ip_Mark[0]))
if((((arp*)ARP_REC_REQ_PTR)->SourceIp[1]&NetPort[num].My_Ip_Mark[1])==(NetPort[num].My_Ip[1]&NetPort[num].My_Ip_Mark[1]))
if((((arp*)ARP_REC_REQ_PTR)->SourceIp[2]&NetPort[num].My_Ip_Mark[2])==(NetPort[num].My_Ip[2]&NetPort[num].My_Ip_Mark[2]))
if((((arp*)ARP_REC_REQ_PTR)->SourceIp[3]&NetPort[num].My_Ip_Mark[3])==(NetPort[num].My_Ip[3]&NetPort[num].My_Ip_Mark[3]))
{//查找有否屬于該IP的對應MAC表
for(i=0;i<MAX_ARP_TERM;i++)
{
if(NetPort[num].ARP_TERM[i].IP_NUM[2]==((arp*)ARP_REC_REQ_PTR)->SourceIp[2])
if(NetPort[num].ARP_TERM[i].IP_NUM[3]==((arp*)ARP_REC_REQ_PTR)->SourceIp[3])
if(NetPort[num].ARP_TERM[i].TTL>0)
{//有則刷新
NetPort[num].ARP_TERM[i].TTL=100;
return(0);
}
}
//查找有否空的MAC表項
for(i=0;i<MAX_ARP_TERM;i++)
{
if(NetPort[num].ARP_TERM[i].TTL==0)
{//有則保存
NetPort[num].ARP_TERM[i].IP_NUM[0]=((arp*)ARP_REC_REQ_PTR)->SourceIp[0];
NetPort[num].ARP_TERM[i].IP_NUM[1]=((arp*)ARP_REC_REQ_PTR)->SourceIp[1];
NetPort[num].ARP_TERM[i].IP_NUM[2]=((arp*)ARP_REC_REQ_PTR)->SourceIp[2];
NetPort[num].ARP_TERM[i].IP_NUM[3]=((arp*)ARP_REC_REQ_PTR)->SourceIp[3];
NetPort[num].ARP_TERM[i].MAC_NUM[0]=((arp*)ARP_REC_REQ_PTR)->SourceMacId[0];
NetPort[num].ARP_TERM[i].MAC_NUM[1]=((arp*)ARP_REC_REQ_PTR)->SourceMacId[1];
NetPort[num].ARP_TERM[i].MAC_NUM[2]=((arp*)ARP_REC_REQ_PTR)->SourceMacId[2];
NetPort[num].ARP_TERM[i].MAC_NUM[3]=((arp*)ARP_REC_REQ_PTR)->SourceMacId[3];
NetPort[num].ARP_TERM[i].MAC_NUM[4]=((arp*)ARP_REC_REQ_PTR)->SourceMacId[4];
NetPort[num].ARP_TERM[i].MAC_NUM[5]=((arp*)ARP_REC_REQ_PTR)->SourceMacId[5];
NetPort[num].ARP_TERM[i].TTL=100;
return(2);
}
}
//MAC表已經滿
return(4);
}
//對方IP即不是本網段也不是GATEWAY
return(1);
}
//===========================================================================
//=======================================================================
/**********************************************************************
**函數原型: void PROCESS_ARP_REC(unsigned char EX_RAM * ARP_PTR)
**入口參數: * ARP_PTR :接收到的ARP包的指針
**出口參數: 無
**返 回 值: 無
**說 明: 對ARP包進行分類處理
************************************************************************/
void PROCESS_ARP_REC(uint8 * ARP_PTR,uint8 num)
{
// EX_RAM PKST ARP_PACKED;
#ifdef Big_End
if(((arp*)ARP_PTR)->Operation==0X0001)
#endif
#ifdef Little_End
if(((arp*)ARP_PTR)->Operation==0X0100)
#endif
{
Arp_Answer(ARP_PTR,num);
}
#ifdef Big_End
else if(((arp*)ARP_PTR)->Operation==0X0002)
#endif
#ifdef Little_End
else if(((arp*)ARP_PTR)->Operation==0X0200)
#endif
{
REC_ARP_REQ(ARP_PTR,num);
}
//可添加REARP操作。
}
/**********************************************************************
**函數原型:? void Arp_Request(unsigned char * ip_address)
**入口參數:?ip_address: 要解析的IP地址
**出口參數: 無
**返 回 值:?無
**說 明:?請求對指定的IP地址進行解析,獲取其物理地址
************************************************************************/
void Arp_Request(uint8 * ip_address,uint8 num)
{
struct _pkst TxdArpReq;
uint8 ARPREQ[46];
uint8 i;
for(i=0;i<6;i++) //復制對方網卡地址或網關地址
{
((arp*)ARPREQ)->SourceMacId[i]=NetPort[num].My_Mac[i];
((arp*)ARPREQ)->DestMacId[i]=0x00; //arp報文的目的物理地址填為0,由arp回答報文 //負責填充
}
for(i=0;i<4;i++)
{
((arp*)ARPREQ)->SourceIp[i]=NetPort[num].My_Ip[i];//填充源IP地址
((arp*)ARPREQ)->DestId[i]=*ip_address;//填充目的IP地址
ip_address++;
}
#ifdef Big_End
((arp*)ARPREQ)->HardwareType=0x0001; //硬件類型:0x0001,以太網類型
((arp*)ARPREQ)->ProtocalType=0x0800; //協議類型:0x0800,對應IPv4
((arp*)ARPREQ)->Operation=0x0001; //操作類型:ARP請求
#endif
#ifdef Little_End
((arp*)ARPREQ)->HardwareType=0x0100; //硬件類型:0x0001,以太網類型
((arp*)ARPREQ)->ProtocalType=0x0008; //協議類型:0x0800,對應IPv4
((arp*)ARPREQ)->Operation=0x0100; //操作類型:ARP請求
#endif
((arp*)ARPREQ)->HardwareLen=0x06; //硬件長度:即物理地址長度,單位字節
((arp*)ARPREQ)->ProtocalLen=0x04; //協議長度:即邏輯地址長度,單位字節
/*******************************************************
*注意:
*arp報文段的長度為28字節,而以太網數據包的最小單元為60字節,所以在發送arp報文時需要對
*以太網報文段進行填充,以滿足最小長度要求.
*arp分組的封裝格式:
*發送報文: | 以太網首部(共14字節) | arp請求分組(28字節) | 填充數據 |
*接收報文: | RTL8019首部 | 以太網首部(共14字節) | arp回答分組(28字節) | 填充數據 |
* |--> 4字節 <--|-------> 14字節 <-------|-------> 28字節 <--------|->18字節<-|
* |--------------------------->0x00~0x<2e<-----------------------------|
*實際上4字節的8019首部在發送時是不起作用的,真正發送的數據從以太網首部起.所以在填充數據時,是從0x2e開始的.
*因為程序中將發送數據報數據結構與接收數據報數據結構定義于公用體中,方便操作.關于數據結構定義
*可以參看arp.h,以及相應的文檔說明.
******************************************************/
//for(i=28;i<29;i++) //不需要46 //當數據長度<60字節時,需要補足60字節數據
// {
ARPREQ[28]=0x00; //填充數據為0x00
// }
//啟動發送數據,發送的是一個arp請求.
TxdArpReq.STPTR=NULL;
TxdArpReq.length=46;
TxdArpReq.DAPTR=ARPREQ;
Send_ethernet_Frame(&TxdArpReq,ARP_REP_MAC,ARP_PACKED,num);
}
/**********************************************************************
**函數原型:? void Initial_arp()
**入口參數:?無
**出口參數: 無
**返 回 值:?無
**說 明:?ARP初始化函數
************************************************************************/
void Initial_arp()
{
uint8 i,num;
Ping_IP_TTL=0;
for(num=0;num<MAX_NET_PORT;num++)
{
NetPort[num].Gateway_IP_TTL=0;
for(i=0;i<MAX_ARP_TERM;i++)
{
NetPort[num].ARP_TERM[i].TTL=0;
}
Arp_Request(NetPort[num].My_Gateway,num);
}
}
?? 快捷鍵說明
復制代碼
Ctrl + C
搜索代碼
Ctrl + F
全屏模式
F11
切換主題
Ctrl + Shift + D
顯示快捷鍵
?
增大字號
Ctrl + =
減小字號
Ctrl + -