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