欧美极品高清xxxxhd,国产日产欧美最新,无码AV国产东京热AV无码,国产精品人与动性XXX,国产传媒亚洲综合一区二区,四库影院永久国产精品,毛片免费免费高清视频,福利所导航夜趣136
標題:
ENC28J60的UDP數據接收
[打印本頁]
作者:
xuwei
時間:
2015-6-13 16:56
標題:
ENC28J60的UDP數據接收
指定雙方IP地址
指定雙方端口號
就可以進行UDP通信了
程序里面的接收文件如下:
*********************************
接收數據包所有程序
************************************/
#include "ARP.h"
#include "ENC28J60_Buff_Operation.h"
#include "ENC28J60_Receive.h"
#include "ENC28J60_SPI.h"
#include "UART.H"
#include "ENC28J60_Send_Receive.h"
uchar flage=0;//釋放FIFO的上部分還是下半部分FLAGE=0,釋放下半部分FLAGE=1釋放上半部分
static xdata uint Receive_Next_Add_Pointer;//下個數據包的開始地址指針
static xdata uint Receive_Now_Add_Pointer;//當前地址指針
static xdata uint Receive_MAC_Type;//數據包的類型
static xdata uint Receive_Size_Packet;//數據包的大小
static xdata uint Receive_Add_Pointer_MSB=0;
static xdata uint Receive_Add_Pointer_LSB=0;
static xdata uint Receive_MAC_Type_MSB; /*接收數據包的臨時地址指針緩沖區*/
static xdata uint Receive_MAC_Type_LSB;
static xdata uint Receive_ARP_Type_MSB;
static xdata uint Receive_ARP_Type_LSB;
static xdata uint Receive_ARP_Type;
xdata uchar Receive_Data[20];//接收到的UDP數據
/*****************************
接收數據包
45頁
*******************************/
void Receiving_data_packet()
{ uchar d2;
Receive_Next_Add_Pointer=0;//緩沖區的數據包下一個地址指針. 初始化
Receive_Now_Add_Pointer=0x300;//當前數據緩沖區地址指針.
///檢測EPKTCNT是否為空
while(1) //死在里面么,這個是故意的
{
write_ENC28J60_REG_SPI(ECON1,5);//開啟接收
for(;;)
{
d2=read_ENC28J60_REG_SPI(EPKTCNT);//有無數據報?
// SendOneByte(d2);
if(d2>0)break;
}
READ_N_Packet();//讀數據包
Check_The_remaining_space();//檢查是否釋放緩沖區
write_ENC28J60_REG_SPI(ECON2,0x40);//PACKET減一
// SendOneByte(Receive_Add_Pointer_MSB);
//SendOneByte(Receive_Add_Pointer_LSB);
//SendOneByte(a);
// SendOneByte(a<<8);
}
}
/*******************************
檢查剩余空間,如果不夠就釋放空間
是這樣的流程:
首先解釋ERXRDPTL和ERXRDPTH這個寄存器是干什么的呢?他就是說硬件通過ERXWRPTL和ERXWRPTH寫入指定的緩存
那么他在那里停止呢?緩存器的底部?錯,這個接收緩沖區就是一個FIFO,他是循環的存儲結構,所以他在到底部
時不會停會跳到開始字節繼續,那么這樣一來他不就停不下了嗎?顯然芯片設計的人不允許他這樣所以設置了這個
寄存器ERXRDPTL和ERXRDPTH,他就是停止一旦硬件寫到他的地址以后的數據他不在接收,全部丟棄,所以要想全部
一個包不少的把以太網上的數據接收回來就要周期性的更新這個地址,也就是說他需要程序自己控制而不要干涉他
否則你將不會得到連續的數據包,我程序里這樣設計的:首先在初始化設定接收緩存的開始地址和結束地址分別是
從0x300-0x1fff這7KB的區域全部是接收緩沖區(FIFO),初始化后ERXWRPTL和ERXWRPTH這個指針自動更新對準0X300
這個寫指針是硬件控制的,用戶控制不了,所以呢不用管,只要在乎這個ERXRDPTL和ERXRDPTH地址指針,這個指針我首先
付給他一個地址0x0e7e,也就是差不多在這個FIFO的一半的位置什么意思呢?表示如果我不更新這個指針的話那么硬件從0X300寫到
0X0E7E就不在寫了,所以我要一邊讀下一個包的地址一邊計算他逼近的距離,當小與400個字節的時候就把這個指針向下移動到0X1FFF
也就是FIFO的底部,這時候硬件就可以繼續向下,但是這時如果不更新這個指針,他就在0X1FFF的時候有就停止了,所以采取同樣的辦法
就是計算逼近距離小于400字節就更新指針到原來的位置0X0E7E,他又可以繼續寫了,就這樣有數據包就收數據包永遠追不上指針,就像小時候
玩的一個游戲一個棍兩頭各一個動物轉動他們他們永遠不相遇,就是這個道理,至于處理數據包這個就被抽象出來了,你想怎么處理就怎么處理
2011年11月23日
無錫第五項目部
王均偉
******************************/
/***********************
檢查是否釋放緩沖區函數
這個函數就是要不斷的檢查接收緩
存數據是不是達到臨界指針,到了
就要馬上讓指針下移以免數據重疊。
*************************/ /**************************/
void Check_The_remaining_space() //讀取當前終止指針的值
{ uint m,l,d,d2; /**************************/
l=read_ENC28J60_REG_SPI(ERXRDPTL);
m=read_ENC28J60_REG_SPI(ERXRDPTH);
d2=((l&0x00ff)+((m<<8)&0xff00)); /*****************************/
d=d2-Receive_Next_Add_Pointer; // 下一個數據包的地址和當前終止地址作比較
//看看是否接近到離終止指針400個字節的地方?
//還有這個Flage表示以0X0E7E為界的緩存的釋放,
//FLAGE=0表示釋放下半部分FIFO(0x0e7e-0x1fff)
//FLAGE=1表示釋放上半部分FIFO(0X1FFF-0X0E7E)
/***************************/
if((d<400)&(flage==0))//如果flage=0和
{
write_ENC28J60_REG_SPI(ERXRDPTL,0xff);
write_ENC28J60_REG_SPI(ERXRDPTH,0x1f);
flage=1;
}
else
if((d<400)&(flage==1))//如果flage=1和空間不足
{
write_ENC28J60_REG_SPI(ERXRDPTL,0x7e);
write_ENC28J60_REG_SPI(ERXRDPTH,0x0e);
flage=0;
}
}
/************************************************
讀取數據包和處理將在這里進行
************************************************/
void READ_N_Packet()
{
/***********************數據處理開始*********************************************/
uint i;
/*定義接收到的包相關信息
比如MAC、源地址IP。目標IP啥的
用來判斷是否需要ARP應答
*/
xdata uchar Receive_MAC_Source_add_MAC[6];
xdata uchar Receive_ARP_Frame_sender_IP[4];
xdata uchar Receive_ARP_Frame_Target_IP[4];
xdata uchar Receive_IP_Frame_Type;//定義IP協議類型
xdata uchar Receive_Source_Port_MSB;
xdata uchar Receive_Source_Port_LSB;
xdata uchar Receive_Destination_MSB;
xdata uchar Receive_Destination_LSB;
xdata uint Receive_Destination;
xdata uint Receive_Source_Port;
xdata uchar Receive_Length_MSB;
xdata uchar Receive_Length_LSB;
xdata uint Receive_Length;//UDP數據長度
//這都是從接收的數據包檢索出的數據,讀出來存變量*/
/**這里解釋一下read_buffer_add(Receive_Now_Add_Pointer+18)為什么是18,
首先抓包后的數據內個MAC層的類型位置在12字節上,而這里要注意,ENC28J60內部
是有個緩存他要先把下一數據包的存放地址寫進去,占兩個字節,然后是4個字節的
狀態字,加起來是6個字節,所以要在ENC28J60的接收緩存中找到MAC幀類型就得要加上6
所以這里是18,注意是ENC28J60,不是PC,詳細應該看手冊的45頁!!!!!
2012年3月1日
于日照高科園
王均偉
**/
Receive_MAC_Type_MSB=read_buffer_add(Receive_Now_Add_Pointer+18); /***讀取數據包的類型*****/
Receive_MAC_Type_LSB=read_buffer_add(Receive_Now_Add_Pointer+19);
Receive_MAC_Type=( ((Receive_MAC_Type_MSB<<8)&0xff00)+(Receive_MAC_Type_LSB&0x00ff));
Receive_IP_Frame_Type=read_buffer_add(Receive_Now_Add_Pointer+29);//讀取IP部首的協議
Receive_Add_Pointer_MSB=read_buffer_add(Receive_Now_Add_Pointer+1);
Receive_Add_Pointer_LSB=read_buffer_add(Receive_Now_Add_Pointer);
Receive_Next_Add_Pointer=(((Receive_Add_Pointer_MSB<<8)&0xff00)+(Receive_Add_Pointer_LSB&0x00ff)); //計算下一個數據包的地址
Receive_ARP_Type_MSB=read_buffer_add(Receive_Now_Add_Pointer+26); /***讀取數據包的操作*****/
Receive_ARP_Type_LSB=read_buffer_add(Receive_Now_Add_Pointer+27);
Receive_ARP_Type=( ((Receive_ARP_Type_MSB<<8)&0xff00)+(Receive_ARP_Type_LSB&0x00ff));
Receive_Size_Packet=(Receive_Next_Add_Pointer-Receive_Now_Add_Pointer);//計算接收到的這個數據包的大小
/*************************************
如果是ARP請求就應答
***************************************/
if((Receive_MAC_Type==0x0806)&(Receive_ARP_Type==0x0001))//檢索ARP請求,回應
{
Receive_MAC_Source_add_MAC[0]=read_buffer_add(Receive_Now_Add_Pointer+12);
Receive_MAC_Source_add_MAC[1]=read_buffer_add(Receive_Now_Add_Pointer+13);
Receive_MAC_Source_add_MAC[2]=read_buffer_add(Receive_Now_Add_Pointer+14);
Receive_MAC_Source_add_MAC[3]=read_buffer_add(Receive_Now_Add_Pointer+15); //發送者MAC
Receive_MAC_Source_add_MAC[4]=read_buffer_add(Receive_Now_Add_Pointer+16);
Receive_MAC_Source_add_MAC[5]=read_buffer_add(Receive_Now_Add_Pointer+17);
Receive_ARP_Frame_sender_IP[0]=read_buffer_add(Receive_Now_Add_Pointer+34);
Receive_ARP_Frame_sender_IP[1]=read_buffer_add(Receive_Now_Add_Pointer+35); //發送者IP
Receive_ARP_Frame_sender_IP[2]=read_buffer_add(Receive_Now_Add_Pointer+36);
Receive_ARP_Frame_sender_IP[3]=read_buffer_add(Receive_Now_Add_Pointer+37);
Receive_ARP_Frame_Target_IP[0]=read_buffer_add(Receive_Now_Add_Pointer+44);
Receive_ARP_Frame_Target_IP[1]=read_buffer_add(Receive_Now_Add_Pointer+45); //目標IP
Receive_ARP_Frame_Target_IP[2]=read_buffer_add(Receive_Now_Add_Pointer+46);
Receive_ARP_Frame_Target_IP[3]=read_buffer_add(Receive_Now_Add_Pointer+47);
/***************************
如果IP地址是本機
******************************/
if((Receive_ARP_Frame_Target_IP[0]==192)&
(Receive_ARP_Frame_Target_IP[1]==168)&
(Receive_ARP_Frame_Target_IP[2]==1) &
(Receive_ARP_Frame_Target_IP[3]==233))
{
/*ARP 應答*/
Send_ARP_Answer_packet( Receive_MAC_Source_add_MAC,Receive_ARP_Frame_sender_IP,Receive_ARP_Frame_Target_IP); //應答
}
}
else
/******************************************************************************
如果不是ARP檢查是不是UDP是就接收數據
*********************************************************************************/
if((Receive_MAC_Type==0x0800)&(Receive_IP_Frame_Type==0x11))//檢索UDP數據,接收
{
/*這個地方用的上滿ARP的變量由于是局部變量所以
可以用,只是用它作為中間量獲取IP,工判斷用
無實際意義*/
Receive_ARP_Frame_sender_IP[0]=read_buffer_add(Receive_Now_Add_Pointer+32);
Receive_ARP_Frame_sender_IP[1]=read_buffer_add(Receive_Now_Add_Pointer+33); //獲得發送者IP
Receive_ARP_Frame_sender_IP[2]=read_buffer_add(Receive_Now_Add_Pointer+34);
Receive_ARP_Frame_sender_IP[3]=read_buffer_add(Receive_Now_Add_Pointer+35);
Receive_ARP_Frame_Target_IP[0]=read_buffer_add(Receive_Now_Add_Pointer+36);
Receive_ARP_Frame_Target_IP[1]=read_buffer_add(Receive_Now_Add_Pointer+37); //獲得目標IP
Receive_ARP_Frame_Target_IP[2]=read_buffer_add(Receive_Now_Add_Pointer+38);
Receive_ARP_Frame_Target_IP[3]=read_buffer_add(Receive_Now_Add_Pointer+39);
Receive_Source_Port_MSB=read_buffer_add(Receive_Now_Add_Pointer+40);
Receive_Source_Port_LSB=read_buffer_add(Receive_Now_Add_Pointer+41); //源端口
Receive_Source_Port=( ((Receive_Source_Port_MSB<<8)&0xff00)+(Receive_Source_Port_LSB&0x00ff));
Receive_Destination_MSB=read_buffer_add(Receive_Now_Add_Pointer+42);
Receive_Destination_LSB=read_buffer_add(Receive_Now_Add_Pointer+43); //目標端口
Receive_Destination=( ((Receive_Destination_MSB<<8)&0xff00)+(Receive_Destination_LSB&0x00ff));
//數據長度
/*如果是192.168.1.89發來*/
if((Receive_ARP_Frame_Target_IP[0]==192)
&&(Receive_ARP_Frame_sender_IP[1]==168)
&&(Receive_ARP_Frame_sender_IP[2]==1)
&&(Receive_ARP_Frame_sender_IP[3]==89)
)
{
/*如果是8080端口發到8080的發來到本機8080的數據接收*/
if( (Receive_Source_Port==8080)
&&(Receive_Destination==8080)
)
{
/*計算接收到的數據字節*/
Receive_Length_MSB=read_buffer_add(Receive_Now_Add_Pointer+44); //
Receive_Length_LSB=read_buffer_add(Receive_Now_Add_Pointer+45); //
Receive_Length=((((Receive_Length_MSB<<8)&0xff00)+(Receive_Length_LSB&0x00ff))-8);
/*讀取數據字節最大20個字節*/
for(i=0;i<Receive_Length;i++)
{
Receive_Data[i]=read_buffer_add(Receive_Now_Add_Pointer+48+i);
}
P0=Receive_Data[0];
}
}
}
/***********************數據處理結束*********************************************/
Receive_Now_Add_Pointer=Receive_Next_Add_Pointer;//下一個數據包地址指針重裝如
}
復制代碼
歡迎光臨 (http://m.raoushi.com/bbs/)
Powered by Discuz! X3.1