UIP是單片機界聯網的一個很好地選擇,移植這個庫有點復雜,首先是第一步,網卡驅動要寫好,使用的網卡芯片為ENC28J60,驅動可以再工程包里面找到
//配置網卡硬件,并設置MAC地址
//返回值:0,正常;1,失敗;
u8 tapdev_init(u8* macaddr)
{
u8 i,res=0;
res=ENC28J60_Init((u8*)macaddr); //初始化ENC28J60
//把IP地址和MAC地址寫入緩存區
for (i = 0; i 《 6; i++)uip_ethaddr.addr[i]=macaddr[i];
//指示燈狀態:0x476 is PHLCON LEDA(綠)=links status, LEDB(紅)=receive/transmit
//PHLCON:PHY 模塊LED 控制寄存器
ENC28J60_PHY_Write(PHLCON,0x0476);
return res;
}
//讀取一包數據
uint16_t tapdev_read(void)
{
return ENC28J60_Packet_Receive(MAX_FRAMELEN,uip_buf);
}
//發送一包數據
void tapdev_send(void)
{
ENC28J60_Packet_Send(uip_len,uip_buf);
}
分別是初始化,讀,寫
這些驅動會在一個叫做uip_call的函數中用到,其次,要設置uip的時鐘,這個時鐘適用于arp表的更新的
#include “clock-arch.h”
#include “sys.h”
//時鐘驅動文件,
//uip時鐘
extern u32 uip_timer;//uip 計時器,每10ms增加1.
/*---------------------------------------------------------------------------*/
clock_time_t
clock_time(void)
{
return uip_timer; /* 10ms 單位 */
}
u32 uip_timer=0;//uip 計時器,每10ms增加1.
//定時器6中斷服務程序
void TIM6_IRQHandler(void)
{ if (TIM_GetITStatus(TIM6, TIM_IT_Update) != RESET) //檢查指定的TIM中斷發生與否:TIM 中斷源
{
uip_timer++;//uip計時器增加1
}
TIM_ClearITPendingBit(TIM6, TIM_IT_Update ); //清除TIMx的中斷待處理位:TIM 中斷源
}
//基本定時器6中斷初始化
//這里時鐘選擇為APB1的2倍,而APB1為36M
//arr:自動重裝值。
//psc:時鐘預分頻數
//這里使用的是定時器3!
void TIM6_Int_Init(u16 arr,u16 psc)
{
TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure;
NVIC_InitTypeDef NVIC_InitStructure;
RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM6, ENABLE); //時鐘使能
TIM_TimeBaseStructure.TIM_Period = arr; //設置在下一個更新事件裝入活動的自動重裝載寄存器周期的值 計數到5000為500ms
TIM_TimeBaseStructure.TIM_Prescaler =psc; //設置用來作為TIMx時鐘頻率除數的預分頻值 10Khz的計數頻率
TIM_TimeBaseStructure.TIM_ClockDivision = 0; //設置時鐘分割:TDTS = Tck_tim
TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up; //TIM向上計數模式
TIM_TimeBaseInit(TIM6, &TIM_TimeBaseStructure); //根據TIM_TimeBaseInitStruct中指定的參數初始化TIMx的時間基數單位
TIM_ITConfig( TIM6,TIM_IT_Update|TIM_IT_Trigger,ENABLE);//使能定時器6更新觸發中斷
TIM_Cmd(TIM6, ENABLE); //使能TIMx外設
NVIC_InitStructure.NVIC_IRQChannel = TIM6_IRQn; //TIM3中斷
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 1; //先占優先級0級
NVIC_InitStructure.NVIC_IRQChannelSubPriority = 2; //從優先級3級
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; //IRQ通道被使能
NVIC_Init(&NVIC_InitStructure); //根據NVIC_InitStruct中指定的參數初始化外設NVIC寄存器
}
定時器的定時長度取決于這個宏定義
#ifndef __CLOCK_ARCH_H__
#define __CLOCK_ARCH_H__
typedef int clock_time_t;
#define CLOCK_CONF_SECOND 100
#endif /* __CLOCK_ARCH_H__ */
上面是100,也就是說定時器的長度應該是10MS
接下來是配置回調函數
//uip事件處理函數
//必須將該函數插入用戶主循環,循環調用。
void uip_polling(void)
{
u8 i;
static struct timer periodic_timer, arp_timer;
static u8 timer_ok=0;
if(timer_ok==0)//僅初始化一次
{
timer_ok = 1;
timer_set(&periodic_timer,CLOCK_SECOND/2); //創建1個0.5秒的定時器
timer_set(&arp_timer,CLOCK_SECOND*10); //創建1個10秒的定時器
}
uip_len=tapdev_read(); //從網絡設備讀取一個IP包,得到數據長度.uip_len在uip.c中定義
if(uip_len》0) //有數據
{
//處理IP數據包(只有校驗通過的IP包才會被接收)
if(BUF-》type == htons(UIP_ETHTYPE_IP))//是否是IP包?
{
uip_arp_ipin(); //去除以太網頭結構,更新ARP表
uip_input(); //IP包處理
//當上面的函數執行后,如果需要發送數據,則全局變量 uip_len 》 0
//需要發送的數據在uip_buf, 長度是uip_len (這是2個全局變量)
if(uip_len》0)//需要回應數據
{
uip_arp_out();//加以太網頭結構,在主動連接時可能要構造ARP請求
tapdev_send();//發送數據到以太網
}
}else if (BUF-》type==htons(UIP_ETHTYPE_ARP))//處理arp報文,是否是ARP請求包?
{
uip_arp_arpin();
//當上面的函數執行后,如果需要發送數據,則全局變量uip_len》0
//需要發送的數據在uip_buf, 長度是uip_len(這是2個全局變量)
if(uip_len》0)tapdev_send();//需要發送數據,則通過tapdev_send發送
}
}else if(timer_expired(&periodic_timer)) //0.5秒定時器超時
{
timer_reset(&periodic_timer); //復位0.5秒定時器
//輪流處理每個TCP連接, UIP_CONNS缺省是40個
for(i=0;i
{
uip_periodic(i); //處理TCP通信事件
//當上面的函數執行后,如果需要發送數據,則全局變量uip_len》0
//需要發送的數據在uip_buf, 長度是uip_len (這是2個全局變量)
if(uip_len》0)
{
uip_arp_out();//加以太網頭結構,在主動連接時可能要構造ARP請求
tapdev_send();//發送數據到以太網
}
}
#if UIP_UDP //UIP_UDP
//輪流處理每個UDP連接, UIP_UDP_CONNS缺省是10個
for(i=0;i
{
uip_udp_periodic(i); //處理UDP通信事件
//當上面的函數執行后,如果需要發送數據,則全局變量uip_len》0
//需要發送的數據在uip_buf, 長度是uip_len (這是2個全局變量)
if(uip_len 》 0)
{
uip_arp_out();//加以太網頭結構,在主動連接時可能要構造ARP請求
tapdev_send();//發送數據到以太網
}
}
#endif
//每隔10秒調用1次ARP定時器函數 用于定期ARP處理,ARP表10秒更新一次,舊的條目會被拋棄
if(timer_expired(&arp_timer))
{
timer_reset(&arp_timer);
uip_arp_timer();
}
}
}
這個函數是uip的靈魂,可以說全部的功能都是在這個函數里面實現的,然后定義網卡數據回調函數
//通信程序狀態字(用戶可以自己定義)
enum
{
STATE_CMD = 0, //命令接收狀態
評論