lwip是瑞典计算机科学院网络嵌入式系统小组(SICS)的Adam Dunkels(亚当·邓克尔) 开发的一个小型开源的TCP/IP协议栈。实现的重点是在保持 TCP 协议主要功能的基础上减少对RAM的占用。
LwIP是Light Weight(轻型)IP 协议,有无操作系统的支持都可以运行。LwIP 实现的重点是在保持TCP协议 主要功能的基础上减少对RAM的占用,它只需十几KB的RAM和 40K左右的ROM就可以运行,这使LwIP协议栈适合在低端的嵌入式系统中使用。
STM32移植LWIP协议栈示例:https://blog.csdn.net/weixin_44453694/article/details/123119624
硬件平台:STM32F103ZE开发板、DM9000有线网卡
开发环境:KEIL5
#include "lwip_config.h"
#include "lwip/tcp.h"
/*接收成功回调函数*/
u8 buff[1024];
u16 rx_len=0;
err_t tcp_recv_func(void *arg, struct tcp_pcb *tpcb,struct pbuf *p, err_t err)
{
memset(buff,0,sizeof(buff));
rx_len=0;
if(p==NULL)
{
clinet_stat=0;
printf("[%d.%d.%d.%d:%d]:客户端断开连接\r\n",(u8)(tpcb->remote_ip.addr),
(u8)(tpcb->remote_ip.addr>>8),
(u8)(tpcb->remote_ip.addr>>16),
(u8)(tpcb->remote_ip.addr>>24),
tpcb->remote_port);
}
else
{
if(p->tot_len==p->len)
{
memcpy(buff,p->payload,p->len);
rx_len=p->len;
pbuf_free(p);
}
else
{
struct pbuf *temp=p;
struct pbuf *q=temp;
while(temp!=NULL)
{
memcpy(buff+rx_len,temp->payload,temp->len);
q=temp;
temp=temp->next;
rx_len+=temp->len;
pbuf_free(q);
}
}
buff[rx_len]='\0';
printf("[%d.%d.%d.%d:%d]:%s\r\n",(u8)(tpcb->remote_ip.addr),
(u8)(tpcb->remote_ip.addr>>8),
(u8)(tpcb->remote_ip.addr>>16),
(u8)(tpcb->remote_ip.addr>>24),
tpcb->remote_port,
buff);
}
return ERR_OK;
}
/*客户端连接成功回调函数*/
u8 client_addr[4];//IP地址
u16 client_prot=0;
u8 clinet_stat=0;
err_t tcp_client(void *arg, struct tcp_pcb *newpcb, err_t err)
{
client_addr[0]=newpcb->remote_ip.addr>>0;
client_addr[1]=newpcb->remote_ip.addr>>8;
client_addr[2]=newpcb->remote_ip.addr>>16;
client_addr[3]=newpcb->remote_ip.addr>>24;
clinet_stat=1;
printf("客户端连接成功:%d.%d.%d.%d:%d\r\n",client_addr[0],client_addr[1],client_addr[2],client_addr[3],newpcb->remote_port);
new_tcp=newpcb;
tcp_recv(newpcb,tcp_recv_func);
return ERR_OK;
}
/*TCP服务器创建*/
struct tcp_pcb *new_tcp;//tcp网络信息(套接字)
u8 LWIP_CreateTcpServer(u16 port)
{
/*1.建立一个新的网卡设备*/
new_tcp=tcp_new();
if(new_tcp==NULL)return 1;
/*2.绑定IP地址和端口号*/
if(tcp_bind(new_tcp, IP_ADDR_ANY,port)!=ERR_OK)
{
return 2;//绑定端口号失败
}
/*开始监听*/
new_tcp=tcp_listen(new_tcp);
/*等待客户端连接*/
tcp_accept(new_tcp,tcp_client);
return 0;
}
#include "dm9000.h"
#include "lwip_config.h"
u8 buff_tx[]="LWIP协议使用示例,发送数据测试示例.";
int main()
{
char buff[200];
u8 stat;
u8 key;
Beep_Init();
Led_Init();
Key_Init();
W25Q64_Init();
Usartx_Init(USART1,115200,72);
TIMx_Init(TIM2,72,20*1000);
IIC_Init();
printf("初始化完成\r\n");
NT35310_Init();
/*DM9000初始化*/
LCD_ShowStr(30,30,16,"DM9000初始化中。。。");//显示字符串
if(DM9000_Init()==0)
{
printf("DM9000初始化成功\r\n");
LCD_ShowStr(30,30+20,16,"DM9000\t OK!");//显示字符串
}
else
{
printf("DM9000初始化失败\r\n");
LCD_ShowStr(30,30+20,16,"DM9000\t ERR!");//显示字符串
}
/*获取DM9000工作模式*/
LCD_ShowStr(128,30+20*2,16,"网卡信息");//显示字符串
stat=DM9000_Get_SpeedAndDuplex();//获取连接状态和工作方式
if(stat!=0xff)
{
printf("网卡速度:%d Mbps 模式:%s\r\n",(stat&0x02)?10:100,(stat&0x01)?"全双工":"半双工");
snprintf(buff,sizeof(buff),"网卡速度:%d MHZ\t %s",(stat&0x02)?10:100,(stat&0x01)?"全双工":"半双工");
LCD_ShowStr(30,30+20*3,16,(u8 *)buff);//网卡速度
}
else
{
printf("DM9000网卡状态信息获取失败!\r\n");
LCD_ShowStr(30,30+20*3,16,(u8 *)"获取网卡信息失败!");//网卡速度
}
LWIP_Config_Init();//LWIP协议栈初始化
while(!lwip_dhcp_stat)//等待IP分配成功
{
LWIP_DataUpdata();
}
TIMx_Init(TIM6,72,1000);
TIM6->CR1|=1<<0;
LWIP_CreateTcpServer(8899);//创建服务器
while(1)
{
LWIP_DataUpdata();
key=Key_Scan();
if(key && clinet_stat)
{
tcp_write(new_tcp,buff_tx,strlen((char *)buff_tx),1);
tcp_output(new_tcp);
}
}
}
1.建立TCP连接函数tcp_new
struct tcp_pcb *tcp_new(void)
函数功能:建立一个新的连接标志(pcb)
形 参:无
返回值: pcb 正常建立了连接标志,返回建立的 pcb
NULL 新的 pcb 内存不可用时
2.绑定IP和端口号tcp_bind
err_t tcp_bind (struct tcp_pcb *pcb, struct ip_addr *ipaddr, u16_t port)
函数功能:绑定本地 IP 地址和端口号
形 参: pcb 准备绑定的连接,类似于 BSD 标准中的 Sockets
Ipaddr 绑定的 IP 地址。如果为 IP_ADDR_ANY,则将连接绑定到所有的本地 IP 地址上
port 绑定的本地端口号。注意:千万不要和其它的应用程序产生冲突
返回值: ERR_OK 正确地绑定了指定的连接
ERR_USE 指定的端口号已经绑定了一个连接,产生了冲突
3.使指定连接进入监听状态tcp_listen
struct tcp_pcb *tcp_listen (struct tcp_pcb *pcb)
函数功能:使指定的连接开始进入监听状态
形 参: pcb 指定将要进入监听状态的连接
返回值: pcb 返回一个新的连接标志 pcb,它作为一个参数传递给将要被分派的函数。这样做的原因是
处于监听状态的连接一般只需要较小的内存,于是函数 tcp_listen()就会收回原始连接的内存,
而重新分配一个较小内存块供处于监听状态的连接使用。
NULL 监听状态的连接的内存块不可用时,返回 NULL。如果这样的话,作为参数传递给函
数tcp_listen()的 pcb 所占用的内存将不能够被分配。
4.等待客户端连接tcp_accept
void tcp_accept(struct tcp_pcb pcb,err_t ( accept)(void *arg,struct tcp_pcb *newpcb,err_t err))
函数功能:指定处于监听状态的连接接通后将要调用的回调函数
形 参: pcb 指定一个处于监听状态的连接
accept 指定连接接通后将要调用的回调函数
返回值:无