您的位置 首页 发布

单片机界联网UIP怎么完成tcp与udp协议

单片机界联网UIP如何实现tcp与udp协议-//配置网卡硬件,并设置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);

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, //指令接纳状况

声明:本文内容来自网络转载或用户投稿,文章版权归原作者和原出处所有。文中观点,不代表本站立场。若有侵权请联系本站删除(kf@86ic.com)https://www.86ic.net/changshang/fabu/346206.html

为您推荐

联系我们

联系我们

在线咨询: QQ交谈

邮箱: kf@86ic.com

关注微信
微信扫一扫关注我们

微信扫一扫关注我们

返回顶部