基于ENC28J60+uIP1.0+STM32的UDP Server实现,以及主动发送数据,几个关键的问题可算整明白了!

news2025/6/8 14:36:32

    ENC28J60,是一款SPI接口的以太网PHY+MAC芯片,实现以太网物理层和MAC层硬件通信。uIP是一个TCP/IP软件协议栈,实现TCP、UDP、ARP、ICMP等网络协议。STM32F103RCT6通过SPI接口与ENC28J60通讯,并移植uIP协议,实现一个小型的UDP服务器。

    严格来说,UDP是面向无连接的对等通讯协议,不存在服务器与客户端之说,这里的服务器只是来形容STM32 SERVER系统的UDP的使用方式,即STM32充当服务器角色,接收同网段的不同IP和端口的UDP数据包,并将数据返回。

    至于ENC28J60+uIP移植,相信不少资料与例子可供参考,比如奋斗嵌入式,正点原子等等,本文不再赘述。这里只说我在移植uIP,调试软件时发现的几个问题。

1、不要TCP,只需要UDP、ARP、ICMP协议,移植哪些文件。

下图是移植uIP后实现上述协议的截图,没错,只要“uip.c”和“uip_arp.c”和必要的头文件就足够了!多一个文件都是浪费!头文件途径也是添加uip和unix文件夹就好了。

2、UDP服务器,接收同网段不同IP和端口,要改uip.c文件的位置,以及怎么用。

打开“uip.c”文件,约1100行左右,#endif /* UIP_UDP_CHECKSUMS */和/* Demultiplex this UDP packet between the UDP "connections". */之间,添加如下代码:

	//UDP SERVER补丁,客户端IP和端口不受限制
	if((uip_udp_conn != 0)
		&&((uip_udp_conn->rport != UDPBUF->srcport)
		 ||!uip_ipaddr_cmp(uip_udp_conn->ripaddr, UDPBUF->srcipaddr)))  	//如果是已经连接并且和接收到的端口号或者IP地址不一致
	{
		uip_udp_remove(uip_udp_conn);											//删除连接
	  uip_udp_conn->rport = UDPBUF->srcport;            //将目的端口设置为收到的远端UDP包的源端口
		uip_udp_conn->lport = UDPBUF->destport;           //将本地端口设置为收到的远端UDP包的目的端口
		memcpy(uip_udp_conn->ripaddr, UDPBUF->srcipaddr, sizeof(uip_ipaddr_t ));	 //将目的IP地址设置为收到的远端UDP包的源IP地址
  }
	
  if(uip_udp_conn->rport == 0)									   		//如果首次接收到某个远端UDP包
	{
    uip_udp_conn->rport = UDPBUF->srcport;            //将目的端口设置为收到的远端UDP包的端口
		memcpy(uip_udp_conn->ripaddr,UDPBUF->srcipaddr,sizeof(uip_ipaddr_t ));	 //将目的IP地址设置为收到的远端UDP包的源IP地址
	}
	
	//uip_udp_conn = uip_udp_new(&ipaddr, HTONS(1000));	//建立到远程ipaddr,端口为1000的连接 
	if(uip_udp_conn != 0)  
	{
		uip_udp_bind(uip_udp_conn, UDPBUF->destport);//绑定本地端口为LPORT,也就是LPORT-->RPORT 发数据 
	}
  //end 补丁

改好后是这样的:

这段代码参考了奋斗的程序,奋斗的程序有BUG,最开始的那个if语句

        if((uip_udp_conn != 0)
        &&((uip_udp_conn->rport != UDPBUF->srcport)
         ||!uip_ipaddr_cmp(uip_udp_conn->ripaddr, UDPBUF->srcipaddr)))      //如果是已经连接并且和接收到的端口号或者IP地址不一致

红的标记的部分,原来的奋斗的程序是“||uip_udp_conn->ripaddr!=UDPBUF->srcipaddr)”,ripaddr和srcipaddr是u16类型的指针地址,判断的是指针的地址是否相等,那么必然不等,所以即使同样的IP地址发来的包都会判断为真跳进if的执行语句中。改正后已经解决。

最后的if语句中:uip_udp_bind(uip_udp_conn, UDPBUF->destport);

目的是将收到的UDP包中,目的端口作为服务器的监听端口,每次都会根据客户端的UDP端口改变,即实现不同的IP、不同的源端口、目的端口发给服务器的UDP包,服务器都能处理后原路返回。如果需要固定服务器的监听端口,只需要将UDPBUF->destport改为固定的端口号即可,例如端口号2000,改为uip_udp_bind(uip_udp_conn, HTONS(2000));

再讲讲怎么用。在main函数初始化ENC28J60和uIP之后,随便监听一个端口就可以了,调用如下函数(NetParam.lUdpPort变量自行定义或替换监听端口号,但是其他端口也能监听)

/**
  * @brief  UDP服务器模式
  * @param  None
  * @retval None
*/
void udp_server_creat(void)
{
	uip_listen(HTONS(NetParam.lUdpPort));
	uip_udp_bind(uip_udp_conn, htons(NetParam.lUdpPort));//绑定本地端口为LPORT
}

3、STM32断电再上电后,PC发给STM32的第一包数据,PC收不到STM32回复的原因。

阿莫有个帖子:为什么uip中udp第一次主动发送数据的时候PC没有收到呢,第二次以后后就正常了 (amobbs.com 阿莫电子论坛 - 东莞阿莫电子网站)

    之前我也被这个问题困扰了一段时间,我做的udp server demo,在用PC端用网络调试助手随便发一个什么数据给STM32 SERVER,SERVER收到后,原路回复上电后收到数据的序号,以及收到数据的字节长度,但是每次下载程序后测试,第一个数据都不回复,第二个才开始回复,序号也是跳过0直接回1。

    原因是:STM32 SERVER端的ARP列表上电后被清空,而电脑端的ARP列表又是上一次缓存了STM32 SERVER的IP和MAC,所以电脑给STM32 SERVER发送UDP包时直接就把UDP发过来了而没有发起ARP,这是一方面。另一方面,void UipPro(void)这个函数,参考奋斗的程序,如下:

/**
  * @brief  中断触发读取网络接收缓存
  * @param  None
  * @retval None
*/
void UipPro(void)
{
	if(ETH_INT == 1){					//当网络接收到数据时,会产生中断
rep:;
		ETH_INT = 0;
		uip_len = tapdev_read();	//从网络设备读取一个IP包,返回数据长度
		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();		//发送数据到以太网(设备驱动程序)
				}
			}
			/* 处理arp报文 */
			else if (BUF->type == htons(UIP_ETHTYPE_ARP))	//是ARP请求包
			{
				uip_arp_arpin();		//如是是ARP回应,更新ARP表;如果是请求,构造回应数据包
				/*
					当上面的函数执行后,如果需要发送数据,则全局变量 uip_len > 0
					需要发送的数据在uip_buf, 长度是uip_len  (这是2个全局变量)
				*/
				if (uip_len > 0)		//是ARP请求,要发送回应
				{
					tapdev_send();		//发ARP回应到以太网上
				}
			}
		}
  }
  else{	  //防止大包造成接收死机,当没有产生中断,而ENC28J60中断信号始终为低说明接收死机
  	 if(ENC28J60_INT_STA == 0) goto rep; 	
  }
}

uip_arp_ipin();           //去除以太网头结构,更新ARP表

这一行本意是,直接从收到的IP包里面获取源IP和源MAC,加入到本地ARP表中,然而全局搜索一下这个函数,发现其在uip_arp.h中是空定义的,上面的函数声明被注释了,

再看uip_arp.c文件,#if 0也是将此函数注释了。


有可能uIP作者调试后忘记将其解除注释,因此,uip_arp_ipin();           //去除以太网头结构,更新ARP表 预期没有实现。

因此,对于STM32来说,它直接收到一个UDP包,要回复给发来这个包的IP地址,但是它的ARP列表是空的,STM32直接就懵了,不知道对方的MAC是啥,于是STM32就放弃发送这包数据,改为发送ARP查询包,询问对方IP。在uip_arp_out()函数中有如下注释就是说了这个意思:

了解原因后,将其释放出来,再编译,发现STM32 SERVER重新上电后第一包数据不回复问题解决。

uIP之uip_arp_ipin();函数的作用。-OpenEdv-开源电子网 这个帖子也是问了这个问题,我没有ID,只能在这里回答了。

4、主动发送UDP数据包的原理及实现。

uIP协议栈用于“收到后发送”的逻辑很好用,但是主动发送要费一番周折,看一遍uip.c这个文件,其发送的大概流程是

uip_process函数会周期调用,flag变量如果为“UIP_UDP_TIMER”状态,就会进入到回调函数UIP_UDP_APPCALL();,然后goto udp_send;进入发送数据流程,如果缓存数据长度变量uip_slen非0,则发送数据。

因此,想要主动发送UDP包,需要让uip_process函数进入UIP_UDP_TIMER流程,然后在回调函数中,注入要发送的数据和长度,之后就发出去了。逻辑就是这么简单,代码如下:

先定义俩个全局变量

u8 *pActiveSendData;  //UDP主动发送,数据指针
u16 ActiveSendLen;    //UDP主动发送数据长度
/**
  * @brief  主动发送udp包
  * @param  data 数据
  * @param  len  数据长度
  * @retval None
*/
void udp_active_send(u8 *data, u16 len)
{
	pActiveSendData = data;
	ActiveSendLen = len;
	
	//设置目标ip和端口
	uip_udp_conn->rport = HTONS(NetParam.rUdpPort);    //将目的端口设置为收到的远端UDP包的源端口
	uip_udp_conn->lport = HTONS(NetParam.lUdpPort);    //将本地端口设置为收到的远端UDP包的目的端口
	uip_ipaddr_t ipaddr;
	uip_ipaddr(ipaddr, NetParam.rIP[0], NetParam.rIP[1], NetParam.rIP[2], NetParam.rIP[3]);	
	memcpy(uip_udp_conn->ripaddr, ipaddr, sizeof(uip_ipaddr_t ));
	
	//唤起UDP发送处理
	uip_process(UIP_UDP_TIMER);
	
	/* 如果上面的函数调用导致数据应该被发送出去,全局变量uip_len设定值> 0 */
	if(uip_len > 0)
	{
		uip_arp_out();		//加以太网头结构,在主动连接时可能要构造ARP请求
		tapdev_send();		//发送数据到以太网(设备驱动程序)
		ActiveSendLen = 0;//发送完成数据清零
	}
}

回调函数向缓冲区写入数据,就是用UIP_UDP_APPCALL()宏定义的那个要自己加的函数

/**
  * @brief  UDP主函数
  * @param  None
  * @retval None
*/
void udp_server_appcall(void)
{
	//接收到一个新的udp数据包
	if(uip_newdata())//收到客户端发过来的数据
	{
		UDP_newdata();
	}
	else if(uip_poll())//主动发送数据
	{
		if(ActiveSendLen)
		{
			udp_send(pActiveSendData, ActiveSendLen);
		}
	}
}

发送函数udp_send

/**
  * @brief  UDP 数据包发送
  * @param  str:数据
  * @retval None
*/
void udp_send(u8 *str, u16 Len)
{
   struct udp_server_appstate *s = (struct udp_server_appstate *)&uip_udp_conn->appstate;

   s->textptr =(u8*)str;
   s->textlen =Len;
   uip_send(s->textptr, s->textlen);//发送udp数据包
//   uip_udp_send(s->textlen);
}

怎么用?主动发送直接调用udp_active_send,如果是先收再发建议还是用udp_send

udp_active_send("OK", 2);

最后附上测试图,1S间隔主动发送

两个PC客户端,不同IP和源目的端口,轮流发送测试。

本例程中ENC28J60与STM32F103RCT6硬件接线:

SPI4根线:CS:PA3   

                  SCK,MISO MOSI :PA5 PA6 PA7

ENC28J60中断  INT:PC4

IP参数定义位于udp_server.c文件中,根据需要自己改,lIP就是STM32 SERVER的IP,rIP是电脑的IP。

最后的最后,放上程序:

链接:https://pan.baidu.com/s/1JmLrI4zj7Bs5ifaHQ3OxAg?pwd=ldp9 
提取码:ldp9

如果链接失效记得评论区喊我更新!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1027049.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

数据分析回头看2——重复值检查/元素替换/异常值筛选

0、前言: 这部分内容是对Pandas的回顾,同时也是对Pandas处理异常数据的一些技巧的总结,不一定全面,只是自己在数据处理当中遇到的问题进行的总结。 1、当数据中有重复行的时候需要检测重复行: 方法:使用p…

BERT: 面向语言理解的深度双向Transformer预训练

参考视频: BERT 论文逐段精读【论文精读】_哔哩哔哩_bilibili 背景 BERT算是NLP里程碑式工作!让语言模型预训练出圈! 使用预训练模型做特征表示的时候一般有两类策略: 1. 基于特征 feature based (Elmo)…

优化系统报错提示信息,提高人机交互(二)

如果服务器接口报错,接口返回报错信息,是怎么实现的呢? 接口调用示例 controller代码 AllArgsConstructor RestController Slf4j public class DemoController {GetMapping("exceptionTest")public Result exceptionTest(Integer…

个人所思所想录

🧑‍💻作者名称:DaenCode 🎤作者简介:CSDN实力新星,后端开发两年经验,曾担任甲方技术代表,业余独自创办智源恩创网络科技工作室。会点点Java相关技术栈、帆软报表、低代码平台快速开…

优化系统报错提示信息,提高人机交互(一)

1、常规报错及处理 package com.example.demo.controller;import com.example.demo.service.IDemoService; import lombok.AllArgsConstructor; import lombok.extern.slf4j.Slf4j; import org.springframework.web.bind.annotation.GetMapping; import org.springframework.w…

13基于PCA的人脸识别,程序已调通,可将自己的数据替换进行识别,得到识别准确率结果,MATLAB平台。

基于PCA的人脸识别,程序已调通,可将自己的数据替换进行识别,得到识别准确率结果,MATLAB平台。

DevSecOps内置安全保护

前言 随着DevOps的发展,DevOps大幅提升了企业应用迭代的速度。但同时,安全如果不能跟上步伐,不仅会抵消DevOps变革带来的提升,拖慢企业数字化转型进程,还会导致漏洞与风险不约而至。所以安全能力在全球范围内受到的重…

2023_Spark_实验十二:Spark高级算子使用

掌握Spark高级算子在代码中的使用 相同点分析 三个函数的共同点,都是Transformation算子。惰性的算子。 不同点分析 map函数是一条数据一条数据的处理,也就是,map的输入参数中要包含一条数据以及其他你需要传的参数。 mapPartitions函数是一个…

IOTE 2023国际物联网展直击:芯与物发布全新定位芯片,助力多领域智能化发展

IOTE 2023国际物联网展,作为全球物联网领域的盛会,于9月20日在中国深圳拉开帷幕。北斗星通集团应邀参展,旗下专业从事物联网、消费类GNSS芯片研发设计的芯与物公司也随其亮相本届盛会。 展会上,芯与物展示了一系列创新的GNSS定位…

基于Android+OpenCV+CNN+Keras的智能手语数字实时翻译——深度学习算法应用(含Python、ipynb工程源码)+数据集(五)

目录 前言总体设计系统整体结构图系统流程图 运行环境模块实现1. 数据预处理2. 数据增强3. 模型构建4. 模型训练及保存5. 模型评估6. 模型测试 系统测试1. 训练准确率2. 测试效果3. 模型应用1)程序下载运行2)应用使用说明3)测试结果 相关其它…

【深度学习】快速部署ONNX模型【入门】

【深度学习】快速部署ONNX模型【入门】 提示:博主取舍了很多大佬的博文并亲测有效,分享笔记邀大家共同学习讨论 文章目录 【深度学习】快速部署ONNX模型【入门】前言搭建打包环境打包exe文件总结 前言 之前的内容已经尽可能简单、详细的介绍CPU【Pytorch2ONNX】和GPU【Pytorch…

编译opencv-3.4.5 [交叉编译]

在unbuntu20.04环境下编译opencv3.4.5, cmake 版本:3.27.4 gcc 版本:11.4.0 g版本:11.4.0 在此环境下编译opencv4.5.4正常。 1. 编译时遇到的问题 (1) Built target libprotobuf make: *** [Makefile:163…

玩玩“小藤”开发者套件 Atlas 200I DK A2 之部署智能语音助手

玩玩“小藤”开发者套件 Atlas 200I DK A2 之部署智能语音助手 0. 背景1. 安装 flac2. 创建自签名证书3. 创建虚拟环境4. 安装PyTorch5. 安装 PyTorch 插件 torch_npu6. 安装APEX混合精度模块7. 安装依赖库8. 使用 gradio 启动智能语音助手9. 访问智能语音助手 0. 背景 总所周…

和逸云 RK3229 如何进入maskrom强刷模式

图中红圈两个点短接以后插usb,就可以进入maskrom模式强刷

【JavaEE】多线程(四)

多线程(四) 在开始讲之前,我们先来回顾回顾前三篇所讲过的内容~ 线程的概念 并发编程,多进程,比较重,频繁创建销毁,开销大 Thread的使用 创建线程 继承Thread实现Runnable继承Thread&#xff…

提交本地项目到GitHub

文章目录 1 下载git1.1 通过homebrew安装Git1.2 通过Xcode安装 2 创建ssh key、配置git3 提交本地项目到GitHub 说明:该博文参考这篇文章和这段视频 1 下载git 1.1 通过homebrew安装Git 1、未安装homebrew,需安装homebrew /usr/bin/ruby -e "$(…

踩坑:Invalid character found in method name. HTTP method names must be tokens

一、原因 在进行本地小程序与服务端请求时,由于加了签名认证,访问接口时报错 Spring boot端 小程序端 二、解决方案 2.1 更改访问路径 将https:更换成http: 示例:https://localhost:8080 改为 http://localhost:8080 2.2其他原因 ssl证书到期了Tomcat的header缓冲区大小不…

使用docker-compose 部署 MySQL8.0

目录 一、拉取MySQL镜像二、创建挂载目录三、添加配置文件my.cnf (没有特殊需求可以跳过)四、编写 docker-compose.yml 文件五、启动容器六、运行后查看启动容器的情况七、连接测试 一、拉取MySQL镜像 我这里使用的是MySQL8.0.18,可以自行选…

pycharm 中package, directory, sources root, resources root的区别

【遇到的问题】 导入yolov5中有utils文件,自己的代码中也有utils文件,使得yolov5中的这部分引用出错了。 【解决方案】 单独建立detection文件夹,把检测相关的都放在这里,yolov5是github上拉取的源码,发现yolov5中fr…

用于设计 CNN 的 7 种不同卷积

一 说明 最近对CNN架构的研究包括许多不同的卷积变体,这让我在阅读这些论文时感到困惑。我认为通过一些更流行的卷积变体的精确定义,效果和用例(在计算机视觉和深度学习中)是值得的。这些变体旨在保存参数计数、增强推理并利用目标…