STM32之LWIP网络通讯设计-下(十五)

STM32F407 系列文章 - ETH-LWIP(十五)


目录

前言

一、软件设计

二、CubeMX实现

1.配置前准备

2.CubeMX配置

1.ETH模块配置

2.时钟模块配置

3.中断模块配置

4.RCC及SYS配置

5.LWIP模块配置

3.生成代码

1.main文件

2.用户层源文件

3.用户层头文件

4.效果演示

三、移植实现

总结


前言

一般对于许多嵌入式系统或单片机,在其资源受限的环境下,要想实现网络通讯,并保证资源的高效利用和稳定的网络通信,我们一般采用一种轻量级的网络协议lwIP。TI公司的STM32芯片一般都会自带一路以太网口,用于网络通讯,但因其内存资源受限,所以都用采用一种小型化、轻量级的lwIP网络协议,只需十几KB的RAM和大约40K的ROM即可运行,既可以在无操作系统环境下工作,也可以与各种操作系统配合使用,使其成为资源受限的嵌入式系统的理想选择。一般市场上所卖的板子都带这一功能的,需准备STM32F407开发板一块和网线一根。


一、软件设计

前面上一篇博文STM32之LWIP网络通讯设计-上(十四)-CSDN博客讲述了对STM32实现LWIP网络通讯的前提性要求介绍,包含使用到的网络协议、MAC内核、PHY驱动芯片、通讯连接示意图、以及硬件电路原理设计图,为网络通讯软件开发提供了设计指导。今天将完成STM32网络通讯的软件设计,实现该软件有有两种方式,第一种方法是通过可视化工具STM32CubeMX完成对lwIP通讯的配置,一键化生成工程代码;第二种方法是先下载lwIP,然后完成lwIP移植到STM32工程项目中,在完成其通讯配置。这里会给出这两种实现方法,但博主更推荐第一种方法,简便快捷。

二、CubeMX实现

‌STM32CubeMX是一个图形化配置工具,主要用于配置STM32微控制器和微处理器。它通过直观的图形用户界面,帮助用户选择STM32 MCU型号、配置引脚、设置系统时钟和外设参数,并生成相应的初始化C代码。这里对CubeMX可视化工具不做详细的介绍,改天会专门写篇文章介绍它。

1.配置前准备

在进行可视化工具CubeMX设置前,需要先了解处理器网络通讯的电路图实现,主要是了解到处理器使用到的IO引脚以及网络驱动芯片,根据上一篇文章介绍网络通讯设计-上(十四)(有提供stm32f407手册数据和YT8512C驱动芯片数据),线连接到处理器的IO引脚PC4、PC5、PG13、PG14、PG11、PC1、PA2、PA7、PD3,如下图所示。

2.CubeMX配置

打开CubeMX工具,如下所示。

1.ETH模块配置

上图,在右边Pinout View上面,根据电气原理图设置stm32相关IO引脚,如上图所示,设置完相关引脚后在上图左边ETH项,打开可以看到配置的相关引脚PC4、PC5、PG13、PG14、PG11、PC1、PA2、PA7、PD3如下所示。

在上图上,我们选择网络模式为RMII,根据YT8512C驱动芯片数据手册,MAC地址、PHY地址、PHY状态寄存器地址、PHY速度、PHY双工状态、以及PHY相关参数设置如下。

并使能Eth中断配置,如下。

根据电气设计图,设置处理器IO引脚PD3为ETH复位引脚,如下所示。

2.时钟模块配置

根据图纸参数,外部时钟为8MHz,处理器最高位168MHz,完成相关设置如下。

3.中断模块配置

4.RCC及SYS配置

5.LWIP模块配置

最后,别忘记使能LWIP模块,完成对其配置,打开第三方插件Middleware,勾选LWIP,里面可以看到LWIP使用的版本号,并完成LWIP的地址、掩码、TCP、BUF等参数的设置,如下所示。

提一句,CubeMX这里面用的LWIP跟外部官网上是一样,因为其开源免费的特性,CubeMX直接将其内嵌在软件里面使用。

在第三方插件Middleware界面上,还可以看到FATFS通用文件系统设置、以及freeRTOS线程设置等等,说明使用CubeMX可视化工具,很方便、功能也挺多。

3.生成代码

上面完成配置引脚、设置系统时钟、以及外设参数设置后,点击GENERATE CODE直接生成代码和工程文件,在保存的路径上面查看。

  

打开工程,查看代码如下。

1.main文件

这里将完成相关GPIO引脚、LWIP、定时器初始化配置,首先重置所有外围设备,初始化Flash接口和Systick,配置系统时钟,初始化所有已配置的外围设备;然后初始化用户UDP网络设置、和用户参数,并启动1毫秒定时器;最后进入主循环,处理网络上的数据,主要为MX_LWIP_Process()函数和UDP_Data_Process()函数完成,其中MX_LWIP_Process函数为CubeMX工具生成的函数,主要作用为处理网络上接收到的数据,传递给注册的接收回调函数UDP_Receive_Callback解析处理。代码示例如下。

/********************************************************************************* @file           : main.c* @brief          : Main program body******************************************************************************* @attention** <h2><center>&copy; Copyright (c) 2025 STMicroelectronics.* All rights reserved.</center></h2>** This software component is licensed by ST under Ultimate Liberty license* SLA0044, the "License"; You may not use this file except in compliance with* the License. You may obtain a copy of the License at:*                             www.st.com/SLA0044********************************************************************************/
/* Includes ------------------------------------------------------------------*/
#include "main.h"
#include "lwip.h"
#include "tim.h"
#include "gpio.h"
/* Private includes ----------------------------------------------------------*/
#include "eth_user.h"
/* Private function prototypes -----------------------------------------------*/
void SystemClock_Config(void);
/*** @brief  The application entry point.* @retval int*/
int main(void)
{/* MCU Configuration--------------------------------------------------------*//* Reset of all peripherals, Initializes the Flash interface and the Systick. */HAL_Init();/* Configure the system clock */SystemClock_Config();/* Initialize all configured peripherals */MX_GPIO_Init();MX_TIM6_Init();MX_LWIP_Init();MX_TIM7_Init();User_UDP_Init();HAL_TIM_Base_Start_IT(&htim6); // 启动1ms定时器/* Infinite loop */while (1){Delay_us(100);MX_LWIP_Process();UDP_Data_Process();}
}
/*** @brief System Clock Configuration* @retval None*/
void SystemClock_Config(void)
{RCC_OscInitTypeDef RCC_OscInitStruct = {0};RCC_ClkInitTypeDef RCC_ClkInitStruct = {0};/** Configure the main internal regulator output voltage*/__HAL_RCC_PWR_CLK_ENABLE();__HAL_PWR_VOLTAGESCALING_CONFIG(PWR_REGULATOR_VOLTAGE_SCALE1);/** Initializes the RCC Oscillators according to the specified parameters* in the RCC_OscInitTypeDef structure.*/RCC_OscInitStruct.OscillatorType = RCC_OSCILLATORTYPE_HSE;RCC_OscInitStruct.HSEState = RCC_HSE_ON;RCC_OscInitStruct.PLL.PLLState = RCC_PLL_ON;RCC_OscInitStruct.PLL.PLLSource = RCC_PLLSOURCE_HSE;RCC_OscInitStruct.PLL.PLLM = 8;RCC_OscInitStruct.PLL.PLLN = 320;RCC_OscInitStruct.PLL.PLLP = RCC_PLLP_DIV2;RCC_OscInitStruct.PLL.PLLQ = 4;if (HAL_RCC_OscConfig(&RCC_OscInitStruct) != HAL_OK){Error_Handler();}/** Initializes the CPU, AHB and APB buses clocks*/RCC_ClkInitStruct.ClockType = RCC_CLOCKTYPE_HCLK|RCC_CLOCKTYPE_SYSCLK|RCC_CLOCKTYPE_PCLK1|RCC_CLOCKTYPE_PCLK2;RCC_ClkInitStruct.SYSCLKSource = RCC_SYSCLKSOURCE_PLLCLK;RCC_ClkInitStruct.AHBCLKDivider = RCC_SYSCLK_DIV1;RCC_ClkInitStruct.APB1CLKDivider = RCC_HCLK_DIV4;RCC_ClkInitStruct.APB2CLKDivider = RCC_HCLK_DIV2;if (HAL_RCC_ClockConfig(&RCC_ClkInitStruct, FLASH_LATENCY_5) != HAL_OK){Error_Handler();}HAL_RCC_MCOConfig(RCC_MCO1, RCC_MCO1SOURCE_PLLCLK, RCC_MCODIV_4);
}

2.用户层源文件

这里将完成网络用户层设置,主要为网络IP初始化设置、以及网络收发数控制,代码示例如下。

#include "eth_user.h"
#include "stm32f4xx_hal.h"
#include "lwip.h"
#include "lwip/igmp.h"#define UDP_PORT     8888 		  // 网络端口
static struct udp_pcb *upcb;    // UDP通讯的控制块对象
static DeviceStatus SendUDPSt;	// UDP通道的通讯状态
UDP_Message RecvUDPMes;
UDP_Message AckUDPMes;
UDP_Message SendUDPMes;
/******************************************************************************* 描述  : 初始化UDP所用的相关数据对象* 参数  : 无* 返回  : 无
******************************************************************************/
void UDP_Data_Init(void)
{memset(&RecvUDPMes, 0, sizeof(UDP_Message));memset(&AckUDPMes, 0, sizeof(UDP_Message));memset(&SendUDPMes, 0, sizeof(UDP_Message));memset(&SendUDPSt, 0, sizeof(DeviceStatus));SendUDPSt.ID = 0x10;
}
/******************************************************************************* 描述  : 接收回调函数编写* 参数  : upcb UDP协议控制块*         p 收到的数据包缓冲区*         addr 接收数据包的远程IP地址 *         port 接收数据包的远程端口* 返回  : 无
******************************************************************************/
static void UDP_Receive_Callback(void *arg, struct udp_pcb *upcb,struct pbuf *p, const ip_addr_t *addr, u16_t port)
{uint8_t udprecvbuf[MAX_UDP_FRAME_LEN] = {0};struct pbuf *ptmp = p;uint32_t data_len = 0;if(upcb != NULL && p != NULL){ip_addr_t remote_source_ip;IP4_ADDR(&remote_source_ip, 192,168,1,113);		if((port == UDP_PORT) && ((addr->addr) == remote_source_ip.addr)) //判断地址和端口从哪里来的{for (ptmp = p; ptmp != NULL; ptmp = ptmp->next)	 //遍历完整个buf链表{				//判断要拷贝到buf中的数据是否大于UDP_DEMO_RX_BUFSIZE的剩余空间,//如果大于的话就只拷贝剩余长度的数据,否则的话就拷贝所有的数据if (ptmp->len <= (MAX_UDP_FRAME_LEN - data_len)){					memcpy(udprecvbuf + data_len, ptmp->payload, ptmp->len);data_len += ptmp->len;}else{					memcpy(udprecvbuf + data_len, ptmp->payload, (MAX_UDP_FRAME_LEN - data_len)); //拷贝数据data_len = MAX_UDP_FRAME_LEN;}//超出UDP最大数据包,则跳出循环if (data_len > MAX_UDP_FRAME_LEN){				break; }}// 初步处理memcpy(&AckUDPMes, udprecvbuf, data_len);SendUDPSt.LastRecvTime = g_1msTick;SendUDPSt.ComSt = COM_ST_PENDING;if(*((uint32_t*)udprecvbuf) != UDP_FRAME_HEAD) { // 同步头校验AckUDPMes.AnsResult = ACK_HEAD_ERROR;}else if(AckUDPMes.Len != (data_len-4)) { // 帧长度校验AckUDPMes.AnsResult = ACK_LEN_ERROR;}else if(RecvUDPMes.CRC16 != CRC16_Calculate((uint8_t*)&RecvUDPMes.SN, RecvUDPMes.Len-4)) { // CRC校验AckUDPMes.AnsResult = ACK_CRC_ERROR;}else SendUDPSt.ComSt = COM_ST_INIT;}}pbuf_free(p); // 释放内存
}/******************************************************************************* 描述  : 创建udp客户端* 参数  : 无* 返回  : 无
******************************************************************************/
void User_UDP_Init(void)
{ip_addr_t local_ip;ip_addr_t remote_ip;err_t err;/* 初始化UDP数据消息 */UDP_Data_Init();// 组播配置//IP4_ADDR(&local_ip, 224,1,1,201);   //配置本地接收组播的地址//IP4_ADDR(&remote_ip, 224,1,1,113);//独播IP4_ADDR(&local_ip, 192,168,1,201);IP4_ADDR(&remote_ip, 192,168,1,113);upcb = udp_new();  // 创建udp控制块if (upcb!=NULL){	upcb->local_port = UDP_PORT;           // 配置本地端口upcb->local_ip.addr = local_ip.addr;   // 配置本地地址upcb->remote_port = UDP_PORT;          // 绑定远程端口upcb->remote_ip.addr = remote_ip.addr; // 绑定远程地址//加入组播//err= igmp_joingroup(IP_ADDR_ANY,&local_ip); //只需要将接收地址放入igmp组,发送的不需要//连接远程服务器IP和端口err= udp_connect(upcb, &remote_ip, UDP_PORT);if(err == ERR_OK){err = udp_bind(upcb,&local_ip,UDP_PORT); //只能收到绑定的本地接收ip地址和端口的数据//err = udp_bind(upcb,IP_ADDR_ANY,UDP_LOCAL_PORT); //可以收到固定端口上任意ip地址的数// 注册接收回调函数,只要接收到数据,这个回调函数会被lwip内核调用udp_recv(upcb, UDP_Receive_Callback, NULL);  // 函数初始化时注册-接收回调函数	}else{//离开组播地址//igmp_leavegroup(IP_ADDR_ANY,&local_ip);// 断开UDP连接udp_remove(upcb);}}
}/******************************************************************************* 描述  : 发送数据* 参数  : (in)pData 发送数据的指针* 返回  : 无
******************************************************************************/
int8_t UDP_Send_Data(struct udp_pcb *upcb, uint8_t *pData, uint16_t len)
{struct pbuf *p;int8_t ret = ERR_BUF;if (upcb != NULL && pData != NULL && len > 0){/* 分配缓冲区空间 */p = pbuf_alloc(PBUF_TRANSPORT, len, PBUF_POOL);if(p != NULL){/* 填充缓冲区数据 */ret = pbuf_take(p, pData, len);if(ret == ERR_OK){/* 发送udp数据 */ret = udp_send(upcb, p);}			/* 释放缓冲区空间 */pbuf_free(p);}}return ret;
}// 处理单项控制命令
CMD_Ack UDP_Single_Ctrl(UDP_Message msg)
{CMD_Ack ret = ACK_OK;if((uint8_t*)&msg != NULL){switch(msg.Param2){case 0:break;case 1:break;case 2:break;default:break;}}else{ret = ACK_ERROR;}return ret;	
}/******************************************************************************* 描述  : 将接收到的UDP数据包解析处理* 参数  : 无* 返回  : 无
******************************************************************************/
void UDP_Msgs_Process(void)
{if(SendUDPSt.ComSt == COM_ST_INIT) {AckUDPMes.AnsResult = UDP_Single_Ctrl(RecvUDPMes);SendUDPSt.ComSt = COM_ST_TRANS;}
}/******************************************************************************
* 描述  : 向以太网上发送UDP数据包* 参数  : 无* 返回  : 无
******************************************************************************/
void UDP_Send_Msgs(void)
{static uint32_t lastime = 0;/* deal ack data */if(SendUDPSt.ComSt != COM_ST_STOP) {AckUDPMes.Head = UDP_FRAME_HEAD;AckUDPMes.Len = UDP_MSG_LEN;AckUDPMes.CRC16 = CRC16_Calculate((uint8_t*)&AckUDPMes.SN, AckUDPMes.Len-4);UDP_Send_Data(upcb, (uint8_t*)&AckUDPMes, UDP_FULL_FRAM_LEN(AckUDPMes.Len));SendUDPSt.ComSt = COM_ST_STOP;memset(&AckUDPMes, 0, sizeof(UDP_Message));}/*d eal heartbeat data */if(g_1msTick - lastime > 1000) {lastime = g_1msTick;SendUDPMes.Head = UDP_FRAME_HEAD;SendUDPMes.Len = UDP_MSG_LEN + 2;SendUDPMes.SN += 1;SendUDPMes.Param1 = SUB_CMD_KEEPALIVE;SendUDPMes.Param2 = RUN_IDLE;SendUDPMes.AnsResult = ACK_PENDING;SendUDPMes.Data[0] = 0xff;SendUDPMes.Data[1] = 0xff;SendUDPMes.CRC16 = CRC16_Calculate((uint8_t*)&SendUDPMes.SN, SendUDPMes.Len-4);UDP_Send_Data(upcb, (uint8_t*)&SendUDPMes, UDP_FULL_FRAM_LEN(SendUDPMes.Len));}
}/******************************************************************************* 描述  : UDP处理流程(处理现有消息,调度发送队列)* 参数  : 无* 返回  : 无
******************************************************************************/
void UDP_Data_Process(void)
{UDP_Msgs_Process();UDP_Send_Msgs();
}

3.用户层头文件

#ifndef __UDP_USER_H
#define __UDP_USER_H#ifdef __cplusplusextern "C" {
#endif#include "stdint.h"
#include "string.h"
#include "crc.h"
#include "udp.h"
#include "timer.h"
#define MAX_UDP_BUF_SIZE 1000
typedef struct 
{uint32_t Head;      // 同步头uint16_t Len;       // 整帧长度(UDP_MSG_LEN+Data长度)uint16_t CRC16;     // CRC校验结果(UDP_MSG_LEN-4+Data长度的数据)uint16_t SN;        // 帧序号uint32_t Param1;    // 参数1uint32_t Param2;    // 参数2uint32_t AnsResult; // 应答结果uint8_t Data[MAX_UDP_BUF_SIZE]; // 数据内容(不能超过1024字节)
}__attribute__((packed)) UDP_Message;#define UDP_FRAME_HEAD 0xC3A53C5A				// 帧头
#define UDP_MSG_LEN 22				          // 帧头长度
#define UDP_FULL_FRAM_LEN(msgLen) ((msgLen)+4)	// 帧头+整帧长度
#define MAX_UDP_FRAME_LEN (4+UDP_MSG_LEN+MAX_UDP_BUF_SIZE)//sizeof(UDP_Message) UDP最大帧长度typedef enum
{ACK_PENDING = 0,        // 等待反馈ACK_OK = 0x10,          // 命令执行结束,且结果正常ACK_ERROR = 0x11,       // 命令执行过程发生异常ACK_WARNING = 0x12,     // 命令执行结束,但结果异常ACK_HEAD_ERROR = 0x20,   // Head校验错误ACK_CRC_ERROR  = 0x21,   // CRC校验错误ACK_LEN_ERROR  = 0x22,   // 长度校验错误ACK_CMD_INVALID = 0x23,  // 命令参数错误,无法执行ACK_RUNNING = 0x22,     // 有其他命令正在执行
}CMD_Ack; // 回应类型}Sub_CMD_Type;typedef enum
{RUN_IDLE 		  = 0x01,			// 空闲RUN_TEST_CTRL	= 0x02,			// 测试RUN_BURN_IN	  = 0x03,		  // 老炼RUN_UPGRADE		= 0x04			// 在线升级
}RunningStType;   // 软件运行状态类型typedef enum
{COM_ST_UNKNOWN 	= 0x00,		// 未知COM_ST_PENDING 	= 0x01,		// 等待回应COM_ST_INIT 	  = 0x02,		// 初步OKCOM_ST_TRANS 	  = 0x03,		// 可传输COM_ST_STOP	 	  = 0x04		// 已停止
}DeviceComStatus; // 设备通讯状态标识void User_UDP_Init(void);
int8_t UDP_Send_Data(struct udp_pcb *upcb, uint8_t *pData, uint16_t len);
void UDP_Msgs_Process(void);
void UDP_Send_Msgs(void);
void UDP_Data_Process(void);
#ifdef __cplusplus
}
#endif#endif

4.效果演示

编写完用户代码后,进行编译运行,连接仿真器在线调式、或者直接烧写到板子中,打开网络监控助手,效果如下。

三、下载移植实现

这里将实现第二种方法,通过下载、移植lwIP到STM32工程项目中,实现网络通讯。

1.lwip下载

点击打开LWIP官网lwIP - 轻量级 TCP/IP 堆栈,显示界面如下所示。在上面既可以下载所需版本lwip,也可以获得相应技术支持。

1.下载方式一

在上面找到下载区,点击进入后,选择我们需要下载的版本,如下所示。这里为了与前面的CubeMX工具上lwip版本保持一致,也选择lwip 2.12版本。

点击上面的版本进行下载时,有时无法响应,这时我们可以采取git方式下,即方式二下载。

2.下载方式二

选择上面的git存储区,点击“lwIP-轻量级 TCPIP堆栈”(lwip.git - lwIP - 轻量级 TCPIP 堆栈),选择对应的版本进行下载。在该选项下面还有一个“lwIP Contrib - 为轻量级 TCP/IP 堆栈提供的代码”选项(contrib.git - lwIP Contrib - 用户例程代码)即官方Contrib提供的demo例程代码,这里选择lwip-contrib-STABLE-2_1_0_RELEASE.tar.gz进行下载(后面需要用到里面的文件)。

 也可以直接点击下载网址进行下载,提供对应版本下载地址如下:

https://git.savannah.nongnu.org/cgit/lwip.git/snapshot/lwip-STABLE-2_1_2_RELEASE.tar.gzicon-default.png?t=O83Ahttps://git.savannah.nongnu.org/cgit/lwip.git/snapshot/lwip-STABLE-2_1_2_RELEASE.tar.gz下载完后,可以看到如下大小的压缩包。

2.lwip文件介绍

对上面下载的压缩包进行解压,得到如下画面。

上图中的lwip2.1.2文件夹包含了许多文件和子文件夹,关于里面的文件我们不需要关心,主要是记录lwIP源码更新、开源软件license、描述lwIP的特点、介绍lwIP源码包的文件目录信息等等,无关紧要。另外还有三个文件夹doc、src、test,其中doc文件夹里面是关于LwIP的一些文档,可以看成是应用和移植LwIP的指南;test文件夹里面是测试LwIP内核性能的源码,将它们和LwIP源码加入到工程中一起编译,调用它们提供的函数,可以获得许多与LwIP内核性能有关的指标;src文件夹是lwIP源码包中最重要的,它是lwIP的内核文件,也是我们移植到工程中的重要文件。

打开src文件夹,如下所示,主要讲解下面5个文件夹。

api文件夹里面装的是NETCONN API和Socket API相关的源文件,只有在操作系统的环境中,才 能被编译。apps文件夹里面装的是应用程序的源文件,包括常见的应用程序,如httpd、mqtt、tftp、sntp、snmp等。core文件夹里面是LwIP的内核源文件,后续会详细讲解。include文件夹里面是LwIP所有模块对应的头文件。netif文件夹里面是与网卡移植有关的文件,这些文件为我们移植网卡提供了模板,我们可以直接 使用。

LwIP内核是由一系列模块组合而成的,这些模块包括:TCP/IP协议栈的各种协议、内存管理模 块、数据包管理模块、网卡管理模块、网卡接口模块、基础功能类模块、API模块。每个模块是由相关的几个源文件和头文件组成的,通过头文件对外声明一些函数、宏、数据类型,使得其它模块可以方便地调用此模块的功能。而构成每个模块的头文件都被组织在了include目录中,而源文件则根据类型被分散地组织在 api、apps、core、netif目录中。下面的子级文件在不作介绍,具体可以查看LwIP的官方说明文档。

3.lwip移植

打开我们stm32项目工程文件,如下。如果没有Middlewares文件夹,则创建它,然后在 Middlewares文件夹下创建了一个名为“lwip”的子文件夹。在“lwip”文件夹下,我们又创建了两个子文件夹:arch和lwip_app。arch文件夹用于存放lwIP系统的配置文件;而lwip_app文件夹则用于存放用户自定义的文件,例如应用程序的源代码等。最后将上述的lwip-STABLE-2_1_2_RELEASE文件夹里面的src文件夹拷贝到lwip文件夹里面。

 

打开stm32工程,并添加 Middlewares/lwip/src/api、Middlewares/l wip/src/core、Middlewares/lwip/src/netif 和 Middlewares/lwip/arch这四个分组,如下所示。

  

在Middlewares/lwip/src/api分组添加src/api 路径下的全部.c文件,Middlewares/lwip/src/core分组添加src/core路径下除ipv6文件夹的全部.c文件,Middlewares/lwip/src/netif分组添加src/netif路径下的ethernet.c文件,如下图所示。

而arch文件夹主要存放lwipopts.h、cc.h、ethernetif.c/h这四个文件都可以在“lwip-contrib-STABLE-2_1_0_RELEASE”文件包下获取。然后再Middlewares/lwip/arch分组添加arch文件夹下的.c文件。

4.添加PHY驱动代码

这里添加的PHY驱动文件代码,就相当于方式一CubeMX生成项目工程下的lwip.c/h和ethernetif.c/h文件的代码,因为CubeMX界面上配置好的PHY驱动参数,生成为lwip.c/h和ethernetif.c/h文件,lwip官方代码是不带驱动代码的,因其不知道你用哪款驱动芯片。添加PHY驱动代码实现步骤如下:

在工程项目路径Drivers\BSP文件夹中,创建“ETHERNET”文件夹,如下所示。

然后在“ETHERNET”文件夹下新建ethernet.c和ethernet.h这两个文件,并将这两个文件添加到项目工程中Drivers/BSP分组下,如下所示。

  

在这两个文件需要完成以太网驱动初始化和MAC的驱动程序,代码如下示例。

5.添加用户代码

将方式一CubeMX实现的用户层代码添加进来,代码跟上面提供的一样,这里不在给出。添加到工程的 Middlewares\lwip\lwip_app\ 路径下,需添加的文件如下所示,包含源文件和头文件。


总结

下面提供的代码,基于STM32F407ZGT芯片编写,可直接在原子开发板上运行,也可运行在各工程项目上,但需要注意各接口以及相应的引脚应和原子开发板上保持一致。

相应的代码链接:单片机STM32F407-Case程序代码例程-CSDN文库

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

AI浪潮下的IT变革之路:机遇、挑战与重塑未来

前言&#xff1a;哈喽&#xff0c;大家好&#xff0c;今天给大家分享一篇文章&#xff01;并提供具体代码帮助大家深入理解&#xff0c;彻底掌握&#xff01;创作不易&#xff0c;如果能帮助到大家或者给大家一些灵感和启发&#xff0c;欢迎收藏关注哦 &#x1f495; 目录 AI浪…

Edge浏览器网页设置深色模式/暗模式

文章目录 需求分析1. 浏览器中的设置——外观——深色。2. 在Edge浏览器的地址栏如下网址&#xff1a;edge://flags/&#xff0c;直接搜索Dark则有内容弹出&#xff0c;将Default更改为Enable即可设置成功。3. 成果 需求 长期对着电脑屏幕&#xff0c;白色实在太刺眼&#xff…

Type-C双屏显示器方案

在数字化时代&#xff0c;高效的信息处理和视觉体验已成为我们日常生活和工作的关键需求。随着科技的进步&#xff0c;一款结合了便携性和高效视觉输出的设备——双屏便携屏&#xff0c;逐渐崭露头角&#xff0c;成为追求高效工作和娱乐体验人群的新宠。本文将深入探讨双屏便携…

Redis 优化秒杀(异步秒杀)

目录 为什么需要异步秒杀 异步优化的核心逻辑是什么&#xff1f; 阻塞队列的特点是什么&#xff1f; Lua脚本在这里的作用是什么&#xff1f; 异步调用创建订单的具体逻辑是什么&#xff1f; 为什么要用代理对象proxy调用createVoucherOrder方法&#xff1f; 对于代码的详细…

机器学习基础-机器学习的常用学习方法

目录 半监督学习的概念 规则学习的概念 基本概念 机器学习里的规则 逻辑规则 规则集 充分性与必要性 冲突消解 命题逻辑 → 命题规则 序贯覆盖 单条规则学习 剪枝优化 强化学习的概念 1. 强化学习对应了四元组 2. 强化学习的目标 强化学习常用马尔可夫决策过程…

国产3D CAD将逐步取代国外软件

在工业软件的关键领域&#xff0c;计算机辅助设计&#xff08;CAD&#xff09;软件对于制造业的重要性不言而喻。近年来&#xff0c;国产 CAD 的发展态势迅猛&#xff0c;展现出巨大的潜力与机遇&#xff0c;正逐步改变着 CAD 市场长期由国外软件主导的格局。 国产CAD发展现状 …

【机器学习】农业 4.0 背后的智慧引擎:机器学习助力精准农事决策

我的个人主页 我的领域&#xff1a;人工智能篇&#xff0c;希望能帮助到大家&#xff01;&#xff01;&#xff01;&#x1f44d;点赞 收藏❤ 在当今数字化浪潮汹涌澎湃之际&#xff0c;农业领域正经历着一场前所未有的深刻变革&#xff0c;大踏步迈向农业 4.0时代。这一时代…

使用Docker模拟PX4固件的无人机用于辅助地面站开发

前言 最近在制作鸿蒙无人机地面站&#xff0c;模仿的是QGroundControl&#xff0c;协议使用mavlink&#xff0c;记录一下本地模拟mavlink协议通过tcp/udp发送 废话不多说直接上命令 1.启动docker的桌面端 启动之后才能使用docker命令来创建容器 docker run --rm -it jonas…

C#调用OpenCvSharp实现图像的开运算和闭运算

对图像同时进行腐蚀和膨胀操作&#xff0c;顺序不同则效果也不同。先腐蚀后膨胀为开运算&#xff0c;能够消除小斑点和细小的突出物、平滑图像以及改善边缘&#xff1b;先膨胀后腐蚀为闭运算&#xff0c;能够去除噪点、填补图像孔洞、连接邻近物体和平滑物体边界。   OpenCvS…

整数和浮点数的存储

整数的存储 整数的存储分为有符号和无符号的整数的存储&#xff0c;整数2进制的表示方法有三种&#xff0c;分别是原码、反码、补码&#xff0c;内存中存储的是补码&#xff0c;反码可以理解为是一个中转站&#xff0c;原码就是直接将数值按照正负形式翻译成的二进制数字 有符…

<论文>时序大模型如何应用于金融领域?

一、摘要 本文介绍2024年的论文《Financial Fine-tuning a Large Time Series Model》&#xff0c;论文探索了主流的时间序列大模型在金融领域的微调应用实践&#xff0c;为时序大模型的领域微调提供了参考。 译文&#xff1a; 大型模型在自然语言处理、图像生成以及近期的时间…

【Linux】深入理解文件系统(超详细)

目录 一.磁盘 1-1 磁盘、服务器、机柜、机房 &#x1f4cc;补充&#xff1a; &#x1f4cc;通常网络中用高低电平&#xff0c;磁盘中用磁化方向来表示。以下是具体说明&#xff1a; &#x1f4cc;如果有一块磁盘要进行销毁该怎么办&#xff1f; 1-2 磁盘存储结构 ​编辑…

UML系列之Rational Rose笔记七:状态图

一、新建状态图 依旧是新建statechart diagram&#xff1b; 二、工作台介绍 接着就是一个状态的开始&#xff1a;开始黑点依旧可以从左边进行拖动放置&#xff1a; 这就是状态的开始&#xff0c;和活动图泳道图是一样的&#xff1b;只能有一个开始&#xff0c;但是可以有多个…

快速上手 INFINI Console 的 TopN 指标功能

背景 在分布式搜索引擎系统&#xff08;如 Easysearch、Elasticsearch 和 OpenSearch&#xff09;中&#xff0c;性能监控至关重要。为了确保系统的高效运行和资源的合理分配&#xff0c;我们通常需要关注一段时间内关键资源的使用情况&#xff0c;特别是索引、节点和分片的内…

springboot vue uniapp 仿小红书 1:1 还原 (含源码演示)

线上预览: 移动端 http://8.146.211.120:8081/ 管理端 http://8.146.211.120:8088/ 小红书凭借优秀的产品体验 和超高人气 目前成为笔记类产品佼佼者 此项目将详细介绍如何使用Vue.js和Spring Boot 集合uniapp 开发一个仿小红书应用&#xff0c;凭借uniapp 可以在h5 小程序 app…

面向对象分析与设计Python版 分析与设计概述

文章目录 一、软件工程概述二、分析与设计概述三、领域模型 一、软件工程概述 高质量软件系统的基本要求 架构性内聚可重用性可维护性可扩展性灵活性 软件开发过程模型&#xff1a;是指根据软件开发项目从开始到结束的一系列步骤和方法&#xff0c;建模为不同的模型。常见的…

3D目标检测数据集——Waymo数据集

Waymo数据集簡介 发布首页&#xff1a;https://waymo.com/open/ 论文&#xff1a;https://openaccess.thecvf.com/content_CVPR_2020/papers/Sun_Scalability_in_Perception_for_Autonomous_Driving_Waymo_Open_Dataset_CVPR_2020_paper.pdf github&#xff1a;https://github.…

[笔记] 使用 Jenkins 实现 CI/CD :从 GitLab 拉取 Java 项目并部署至 Windows Server

随着软件开发节奏的加快&#xff0c;持续集成&#xff08;CI&#xff09;和持续部署&#xff08;CD&#xff09;已经成为确保软件质量和加速产品发布的不可或缺的部分。Jenkins作为一款广泛使用的开源自动化服务器&#xff0c;为开发者提供了一个强大的平台来实施这些实践。然而…

基于“大型园区”网络设计

基于“大型园区”网络设计 目 录 第1章 项目概述1 1.1 项目背景1 1.2 公司概况1 1.3 网络现状2 第2章 需求分析4 2.1 部门需求4 2.2 配置需求4 2.3 网络功能需求5 第3章 网络设计6 3.1 建设原则6 3.2 网络拓扑结构6 3.3 IP地址和VLAN划分8 3.4 核心层设计9 3.5 …

回归预测 | MATLAB实RVM-Adaboost相关向量机集成学习多输入单输出回归预测

回归预测 | MATLAB实RVM-Adaboost相关向量机集成学习多输入单输出回归预测 目录 回归预测 | MATLAB实RVM-Adaboost相关向量机集成学习多输入单输出回归预测预测效果基本介绍程序设计参考资料 预测效果 基本介绍 RVM-Adaboost相关向量机集成学习多输入单输出回归预测是一种先进…