当前位置:首页 > 单片机 > 单片机
[导读]0.前言 去年(2013年)的整理了LwIP相关代码,并在STM32上“裸奔”成功。一直没有时间深入整理,在这里借博文整理总结。LwIP的移植过程细节很多,博文也不可能一一详解个别部分只能点到为止。 【本文要点】 【1】

0.前言

去年(2013年)的整理了LwIP相关代码,并在STM32上“裸奔”成功。一直没有时间深入整理,在这里借博文整理总结。LwIP的移植过程细节很多,博文也不可能一一详解个别部分只能点到为止。

【本文要点】

【1】不带操作系统的LwIP移植,LwIP版本为1.4.1。

【2】MCU为STM32F103VE,网卡为ENC28J60。

【3】移植过程重点描述ethernetif.c和LwIP宏配置等。

【4】一个简单的TCP echo例子。

【5】力求简单,没有DHCP功能,甚至没有用到网卡中断。

【代码仓库】

代码仓库位于Bitbucket(要源代码请点击这里)。博文中不能把每个细节描述清楚,更多内容请参考代码仓库中的具体代码。

【硬件说明】

测试平台使用奋斗版,原理图请参考代码仓库中的DOC目录。


【参考博文】

学习嵌入式网络是一个循序渐进的过程,从浅入深从简单到复杂。

【1】ENC28J60学习笔记——学习网卡

【2】STM32NET学习笔记——索引——理解TCPIP协议栈

【3】uIP学习笔记——初次应用协议栈

【4】Yeelink平台使用——远程控制 RT Thread + LwIP+ STM32——更加实用的做法

1.ethernetif.c的相关修改

虽然LwIP移植过程比较复杂,但是只要结合网卡具体功能,耐心修改ethernetif.c即可。ethernetif.c重点实现网卡的三个功能,初始化,发送和接收。

为了更好的配合lwIP,修改了ENC28J60学习笔记中部分驱动函数。(换句话说,想要从0开始移植LwIP必须对操作网卡非常熟悉)

【1】初始化


staticvoid

low_level_init(structnetif*netif)

{

structethernetif*ethernetif=netif->state;

/*setMAChardwareaddresslength*/

netif->hwaddr_len=ETHARP_HWADDR_LEN;

/*setMAChardwareaddress*/

netif->hwaddr[0]='A';

netif->hwaddr[1]='R';

netif->hwaddr[2]='M';

netif->hwaddr[3]='N';

netif->hwaddr[4]='E';

netif->hwaddr[5]='T';

/*maximumtransferunit*/

netif->mtu=1500;

/*devicecapabilities*/

/*don'tsetNETIF_FLAG_ETHARPifthisdeviceisnotanethernetone*/

netif->flags=NETIF_FLAG_BROADCAST|NETIF_FLAG_ETHARP|NETIF_FLAG_LINK_UP;

/*Dowhateverelseisneededtoinitializeinterface.*/

enc28j60_init(netif->hwaddr);//【1】

}

【说明】

【1】enc28j60_init(netif->hwaddr); low_level_init中指定了enc28j60中的网卡地址。

【2】发送


staticerr_t

low_level_output(structnetif*netif,structpbuf*p)

{

structethernetif*ethernetif=netif->state;

structpbuf*q;

enc28j60_init_send(p->tot_len);//【1】initiatetransfer();

#ifETH_PAD_SIZE

pbuf_header(p,-ETH_PAD_SIZE);/*dropthepaddingword*/

#endif

for(q=p;q!=NULL;q=q->next){

/*Sendthedatafromthepbuftotheinterface,onepbufata

time.Thesizeofthedataineachpbufiskeptinthe->len

variable.*/

enc28j60_writebuf(q->payload,q->len);//【2】senddatafrom(q->payload,q->len);

}

enc28j60_start_send();//【3】signalthatpacketshouldbesent();

#ifETH_PAD_SIZE

pbuf_header(p,ETH_PAD_SIZE);/*reclaimthepaddingword*/

#endif

LINK_STATS_INC(link.xmit);

returnERR_OK;

}

【说明】

【1】enc28j60_init_send(p->tot_len);初始化发送缓冲区大小,pbuf结构为一个链表,第一个pbuf结构体中的tot_len字段代表整个以太网数据包的大小。

【2】enc28j60_writebuf( q->payload, q->len ); 通过遍历链表把内容填入ENC28J60的缓冲区中。

【3】enc28j60_start_send();启动网卡发送。

【3】接收


staticstructpbuf*

low_level_input(structnetif*netif)

{

structethernetif*ethernetif=netif->state;

structpbuf*p,*q;

u16_tlen;

len=enc28j60_packet_getlen();//【1】

#ifETH_PAD_SIZE

len+=ETH_PAD_SIZE;/*allowroomforEthernetpadding*/

#endif

/*Weallocateapbufchainofpbufsfromthepool.*/

p=pbuf_alloc(PBUF_RAW,len,PBUF_POOL);

if(p!=NULL){

#ifETH_PAD_SIZE

pbuf_header(p,-ETH_PAD_SIZE);/*dropthepaddingword*/

#endif

for(q=p;q!=NULL;q=q->next){

enc28j60_readbuf(q->payload,q->len);//【2】readdatainto(q->payload,q->len);

}

enc28j60_finish_receive();//【3】acknowledgethatpackethasbeenread();

#ifETH_PAD_SIZE

pbuf_header(p,ETH_PAD_SIZE);/*reclaimthepaddingword*/

#endif

LINK_STATS_INC(link.recv);

}else{

enc28j60_finish_receive();//【4】droppacket();

LINK_STATS_INC(link.memerr);

LINK_STATS_INC(link.drop);

}

returnp;

}

【说明】

【1】len = enc28j60_packet_getlen(); 获得网卡中数据包的长度。

【2】enc28j60_readbuf (q->payload, q->len);把网卡中的内容复制到内存池中。

【3】enc28j60_finish_receive();接收完成,移动网卡中缓冲区指针。

【4】应用

【1】LwIP网卡硬件初始化调用ethernetif_init即可,该函数中调用了low_level_init,并指定了网卡输出函数low_level_output。

【2】一旦网卡有数据进入,应立即代用ethernetif_input函数。可以使用中断方法或查询方法。

2.lwipopt.h配置简述

lwip中的配置选项非常的多,了解所有的配置非常不容易。本博文参考STM32官方的两个例子总结得到。


#ifndef__LWIPOPTS_H__

#define__LWIPOPTS_H__

#defineSYS_LIGHTWEIGHT_PROT0

#defineNO_SYS1

#defineNO_SYS_NO_TIMERS1

/*----------Memoryoptions----------*/

/*MEM_ALIGNMENT:shouldbesettothealignmentoftheCPUforwhich

lwIPiscompiled.4bytealignment->defineMEM_ALIGNMENTto4,2

bytealignment->defineMEM_ALIGNMENTto2.*/

#defineMEM_ALIGNMENT4

/*MEM_SIZE:thesizeoftheheapmemory.Iftheapplicationwillsend

a lot of data that needs to be copied, this

本站声明: 本文章由作者或相关机构授权发布,目的在于传递更多信息,并不代表本站赞同其观点,本站亦不保证或承诺内容真实性等。需要转载请联系该专栏作者,如若文章内容侵犯您的权益,请及时联系本站删除( 邮箱:macysun@21ic.com )。
换一批
延伸阅读

在工业控制、音频处理等高性能嵌入式场景中,某电机驱动项目通过混合使用寄存器操作与CMSIS-DSP库,将PID控制周期从120μs缩短至38μs,系统响应速度提升3倍。本文将揭秘这种"底层+高层"混合编程模式的核心技巧。

关键字: HAL STM32 寄存器

嵌入式系统开发手势识别作为非接触式人机交互的核心技术,正从实验室走向消费级应用。然而,传感器采集的原始信号常因电磁干扰、电源噪声或机械抖动产生失真,导致识别准确率下降。本文以STM32微控制器与PAJ7620手势识别传感...

关键字: STM32 手势识别 噪声

在物联网设备开发中,快速实现稳定可靠的网络通信是项目成功的关键。W5500作为一款集成硬件TCP/IP协议栈的以太网控制器,凭借其"开箱即用"的特性,可大幅缩短STM32平台的网络功能开发周期。本文通...

关键字: STM32 W5500

工业物联网设备开发中,某智能电表项目曾因ADC采样中断响应延迟导致数据丢失率高达15%。技术人员通过重构DMA驱动架构,将数据搬运效率提升12倍,CPU占用率从38%降至3%,成功解决高速采样场景下的实时性难题。这一案例...

关键字: STM32 DMA

工业机器人关节控制系统中,一个典型的伺服驱动器需要在100μs周期内完成电流采样、位置反馈、PID计算和PWM输出等12项关键任务。当传统固定优先级调度导致机械臂出现0.3°的位置抖动时,某运动控制厂商通过引入混合排序算...

关键字: 电机控制 STM32

在STM32嵌入式系统开发中,排序算法的效率直接影响传感器数据处理、通信协议解析等核心任务的实时性。传统快速排序在部分有序数据场景下易退化为O(n²)时间复杂度,而单纯依赖三数取中法优化基准值选择仍存在小规模数据效率不足...

关键字: STM32 传感器

在智慧农业的广阔田野里,部署着数百个土壤湿度传感器节点。这些节点通过LoRa模块将数据传输至网关,再由网关上传至云端进行分析。然而,当暴雨来临前,土壤湿度骤增的紧急数据若淹没在常规监测数据的洪流中,可能导致灌溉系统未能及...

关键字: STM32 无线通信 LoRa

在电池管理系统(BMS)中,电压均衡是保障电池组性能与寿命的核心技术。由于电池单体存在制造差异,串联使用过程中易出现电压不一致现象,导致部分电池过充/过放,加速老化。传统被动均衡通过能耗电阻消耗高电压单体的能量,但存在效...

关键字: BMS系统 STM32

在嵌入式系统中,模数转换器(ADC)是连接物理世界与数字处理的核心桥梁。STM32系列微控制器内置的ADC采用逐次逼近型(SAR)架构,通过精密的硬件电路实现模拟信号到数字信号的转换。

关键字: ADC STM32

STM32系列微控制器因其高性能和丰富的外设接口被广泛应用于各类场景。当涉及USB高速(HS)与全速(FS)接口设计时,开发者常因对信号完整性、ESD防护及电源管理的理解不足而陷入调试困境。本文将从原理出发,结合实际案例...

关键字: STM32 USB
关闭