STM32F4XX LWIP+freeRTOS移植(二)

有疑问请加扣扣技术交流群:460189483

源码下载地址:https://download.csdn.net/download/u014453443/10698059

前面几个文件改好了之后,下面进入实战环节,从ucos_ii系统+lwip移植为freeRTOS+LWIP!

第一步:从原子例程中找一个现成的freeRTOS系统工程,这里我们选的FreeRTOS实验20-1 FreeRTOS内存管理实验,如下:

打开之后编译一下,确保工程没有问题!

1.1 然后将LWIP例程中的-->网络实验2 LWIP带UCOSII操作系统移植-->LWIP拷贝到freeRTOS工程下

1.2 然后将LWIP例程中的-->网络实验2 LWIP带UCOSII操作系统移植-->HARDWARE-->LAN8720拷贝到freeRTOS工程中的HARDWARE文件夹下

1.3 在工程上右键 manage project item,然后新增LWIP_APP、LWIP_CORE、LWIP_NETIF、LWIP_API、LWIP_ARCH,如下图所示:

1.4  将相应的文件添加进去,妈的,CSDN没法上传图片了?一直上传不成功

1.5 在main函数中添加头文件,否则系统默认没有定义LWIP_DHCP,系统是不会进行DHCP功能的,头文件如下:

#include "lan8720.h"
#include "string.h"
#include "malloc.h"
#include "FreeRTOS.h"
#include "task.h"
#include "timer.h"
#include "lwip/netif.h"
#include "lwip_comm.h"
#include "lwipopts.h"
#include "tcp_server_demo.h"

1.6 main起始任务中添加如下:lwip_comm_init()一定要放在起始任务中,如果放在main函数中,则因为没有开启freeRTOS系统,会始终初始化不成功,在ucos_ii系统中因为有OSInit()初始化系统函数,所以放在main中没有问题,这一点要注意,否则启动之后始终会卡死在某一个位置!

int main(void)
{
    Stm32_Clock_Init(360,25,2,8);   //设置时钟,180Mhz   
    HAL_Init();                     //初始化HAL库
    delay_init(180);                //初始化延时函数
    uart_init(115200);              //初始化USART
    //usmart_dev.init(90); 		    //初始化USMART	
    LED_Init();                     //初始化LED 
    KEY_Init();                     //初始化按键
    SDRAM_Init();                   //初始化SDRAM
    LCD_Init();                     //初始化LCD
    PCF8574_Init();                 //初始化PCF8574
    my_mem_init(SRAMIN);		    //初始化内部内存池
	my_mem_init(SRAMEX);		    //初始化外部内存池
	my_mem_init(SRAMCCM);		    //初始化CCM内存池
	POINT_COLOR = RED; 		        //红色字体
	LCD_ShowString(30,30,200,20,16,"Apollo STM32F4/F7");
	LCD_ShowString(30,50,200,20,16,"LWIP+UCOSII Test");
	LCD_ShowString(30,70,200,20,16,"ATOM@ALIENTEK");
	LCD_ShowString(30,90,200,20,16,"2016/1/14");	
	
    //创建开始任务
    xTaskCreate((TaskFunction_t )start_task,            //任务函数
                (const char*    )"start_task",          //任务名称
                (uint16_t       )START_STK_SIZE,        //任务堆栈大小
                (void*          )NULL,                  //传递给任务函数的参数
                (UBaseType_t    )START_TASK_PRIO,       //任务优先级
                (TaskHandle_t*  )&StartTask_Handler);   //任务句柄                
    vTaskStartScheduler();          //开启任务调度
}


//开始任务任务函数
void start_task(void *pvParameters)
{
	while(lwip_comm_init()) 	    //lwip初始化
	{
		LCD_ShowString(30,110,200,20,16,"Lwip Init failed!"); 	//lwip初始化失败
		delay_ms(500);
		LCD_Fill(30,110,230,150,WHITE);
		delay_ms(500);
	}
	LCD_ShowString(30,110,200,20,16,"Lwip Init Success!"); 		//lwip初始化成功
	
	while(tcp_server_init()) 									//初始化tcp_server(创建tcp_server程)
	{
		LCD_ShowString(30,110,200,20,16,"TCP Server failed! "); 
		delay_ms(500);
		LCD_Fill(30,110,230,170,WHITE);
		delay_ms(500);
	}
	
    taskENTER_CRITICAL();           //进入临界区
	
	#if LWIP_DHCP
	lwip_comm_dhcp_creat(); //创建DHCP任务
	#endif
    //创建TASK1任务
    xTaskCreate((TaskFunction_t )display_task,             
                (const char*    )"display_task",           
                (uint16_t       )DISPALY_STK_SIZE,        
                (void*          )NULL,                  
                (UBaseType_t    )DISPALY_TASK_PRIO,        
                (TaskHandle_t*  )&DisplayTask_Handler);   
				
				//创建TASK1任务
    xTaskCreate((TaskFunction_t )led_task,             
                (const char*    )"led_task",           
                (uint16_t       )LED_STK_SIZE,        
                (void*          )NULL,                  
                (UBaseType_t    )LED_TASK_PRIO,        
                (TaskHandle_t*  )&LedTask_Handler); 
    vTaskDelete(StartTask_Handler); //删除开始任务
    taskEXIT_CRITICAL();            //退出临界区
}

第二步:修改lwip_common.c函数

2.1 定义LWIP TaskHandle_t的DHCP任务句柄,并删除ucos_ii的堆栈

2.2 修改lwip_comm_mem_malloc函数删除没用的内存申请

3.3 修改lwip_comm_dhcp_creat、lwip_comm_dhcp_delete函数为freeRTOS的API函数,整个文件如下:

__lwip_dev lwipdev;						//lwip控制结构体 
struct netif lwip_netif;				//定义一个全局的网络接口

extern u32 memp_get_memorysize(void);	//在memp.c里面定义
extern u8_t *memp_memory;				//在memp.c里面定义.
extern u8_t *ram_heap;					//在mem.c里面定义.
TaskHandle_t LWIP_DHCP_TaskHandler;
/
//lwip两个任务定义(内核任务和DHCP任务)

//lwip DHCP任务
//设置任务优先级
#define LWIP_DHCP_TASK_PRIO       		3
//设置任务堆栈大小
#define LWIP_DHCP_STK_SIZE  		    256
//任务函数
void lwip_dhcp_task(void *pdata); 

//用于以太网中断调用
void lwip_pkt_handle(void)
{
	ethernetif_input(&lwip_netif);
}

//lwip中mem和memp的内存申请
//返回值:0,成功;
//    其他,失败
u8 lwip_comm_mem_malloc(void)
{
	u32 mempsize;
	u32 ramheapsize; 
    INTX_DISABLE();//关中断
	mempsize=memp_get_memorysize();			//得到memp_memory数组大小
	memp_memory=mymalloc(SRAMIN,mempsize);	//为memp_memory申请内存
	ramheapsize=LWIP_MEM_ALIGN_SIZE(MEM_SIZE)+2*LWIP_MEM_ALIGN_SIZE(4*3)+MEM_ALIGNMENT;//得到ram heap大小
	ram_heap=mymalloc(SRAMIN,ramheapsize);	//为ram_heap申请内存     
    INTX_ENABLE();//开中断
	if(!memp_memory||!ram_heap)//有申请失败的
	{
		lwip_comm_mem_free();
		return 1;
	}
	return 0;	
}
//lwip中mem和memp内存释放
void lwip_comm_mem_free(void)
{ 	
	myfree(SRAMIN,memp_memory);
	myfree(SRAMIN,ram_heap);
}
//lwip 默认IP设置
//lwipx:lwip控制结构体指针
void lwip_comm_default_ip_set(__lwip_dev *lwipx)
{
	u32 sn0;
	sn0=*(vu32*)(0x1FFF7A10);//获取STM32的唯一ID的前24位作为MAC地址后三字节
	//默认远端IP为:192.168.1.100
	lwipx->remoteip[0]=192;	
	lwipx->remoteip[1]=168;
	lwipx->remoteip[2]=1;
	lwipx->remoteip[3]=104;
	//MAC地址设置(高三字节固定为:2.0.0,低三字节用STM32唯一ID)
	lwipx->mac[0]=2;//高三字节(IEEE称之为组织唯一ID,OUI)地址固定为:2.0.0
	lwipx->mac[1]=0;
	lwipx->mac[2]=0;
	lwipx->mac[3]=(sn0>>16)&0XFF;//低三字节用STM32的唯一ID
	lwipx->mac[4]=(sn0>>8)&0XFFF;
	lwipx->mac[5]=sn0&0XFF; 
	//默认本地IP为:192.168.1.30
	lwipx->ip[0]=192;	
	lwipx->ip[1]=168;
	lwipx->ip[2]=1;
	lwipx->ip[3]=30;
	//默认子网掩码:255.255.255.0
	lwipx->netmask[0]=255;	
	lwipx->netmask[1]=255;
	lwipx->netmask[2]=255;
	lwipx->netmask[3]=0;
	//默认网关:192.168.1.1
	lwipx->gateway[0]=192;	
	lwipx->gateway[1]=168;
	lwipx->gateway[2]=1;
	lwipx->gateway[3]=1;	
	lwipx->dhcpstatus=0;//没有DHCP	
} 

//LWIP初始化(LWIP启动的时候使用)
//返回值:0,成功
//      1,内存错误
//      2,LAN8720初始化失败
//      3,网卡添加失败.
u8 lwip_comm_init(void)
{
    u8 retry=0;
	struct netif *Netif_Init_Flag;		//调用netif_add()函数时的返回值,用于判断网络初始化是否成功
	struct ip_addr ipaddr;  			//ip地址
	struct ip_addr netmask; 			//子网掩码
	struct ip_addr gw;      			//默认网关 

	if(ETH_Mem_Malloc())return 1;		//内存申请失败
	if(lwip_comm_mem_malloc())return 2;	//内存申请失败
    lwip_comm_default_ip_set(&lwipdev);	//设置默认IP等信息
	while(LAN8720_Init())		        //初始化LAN8720,如果失败的话就重试5次
    {
        retry++;
        if(retry>5) {retry=0;return 3;} //LAN8720初始化失败
    }
	tcpip_init(NULL,NULL);				//初始化tcp ip内核,该函数里面会创建tcpip_thread内核任务

#if LWIP_DHCP		//使用动态IP
	ipaddr.addr = 0;
	netmask.addr = 0;
	gw.addr = 0;
#else				//使用静态IP
	IP4_ADDR(&ipaddr,lwipdev.ip[0],lwipdev.ip[1],lwipdev.ip[2],lwipdev.ip[3]);
	IP4_ADDR(&netmask,lwipdev.netmask[0],lwipdev.netmask[1] ,lwipdev.netmask[2],lwipdev.netmask[3]);
	IP4_ADDR(&gw,lwipdev.gateway[0],lwipdev.gateway[1],lwipdev.gateway[2],lwipdev.gateway[3]);
	printf("网卡en的MAC地址为:................%d.%d.%d.%d.%d.%d\r\n",lwipdev.mac[0],lwipdev.mac[1],lwipdev.mac[2],lwipdev.mac[3],lwipdev.mac[4],lwipdev.mac[5]);
	printf("静态IP地址........................%d.%d.%d.%d\r\n",lwipdev.ip[0],lwipdev.ip[1],lwipdev.ip[2],lwipdev.ip[3]);
	printf("子网掩码..........................%d.%d.%d.%d\r\n",lwipdev.netmask[0],lwipdev.netmask[1],lwipdev.netmask[2],lwipdev.netmask[3]);
	printf("默认网关..........................%d.%d.%d.%d\r\n",lwipdev.gateway[0],lwipdev.gateway[1],lwipdev.gateway[2],lwipdev.gateway[3]);
#endif
	Netif_Init_Flag=netif_add(&lwip_netif,&ipaddr,&netmask,&gw,NULL,&ethernetif_init,&tcpip_input);//向网卡列表中添加一个网口
	if(Netif_Init_Flag==NULL)return 4;//网卡添加失败 
	else//网口添加成功后,设置netif为默认值,并且打开netif网口
	{
		netif_set_default(&lwip_netif); //设置netif为默认网口
		netif_set_up(&lwip_netif);		//打开netif网口
	}
	return 0;//操作OK.
}   
//如果使能了DHCP
#if LWIP_DHCP
//创建DHCP任务
void lwip_comm_dhcp_creat(void)
{
	taskENTER_CRITICAL();  //进入临界区
	xTaskCreate((TaskFunction_t)lwip_dhcp_task,
						(const char*  )"DHCP_TASK",
						(uint16_t     )LWIP_DHCP_STK_SIZE,
						(void*        )NULL,
						(UBaseType_t  )LWIP_DHCP_TASK_PRIO,
						(TaskHandle_t*)&LWIP_DHCP_TaskHandler);//创建DHCP任务 						
	taskEXIT_CRITICAL();  //退出临界区
}
//删除DHCP任务
void lwip_comm_dhcp_delete(void)
{
	dhcp_stop(&lwip_netif); 		//关闭DHCP
	vTaskDelete(LWIP_DHCP_TaskHandler);	//删除DHCP任务
}
//DHCP处理任务
void lwip_dhcp_task(void *pdata)
{
	u32 ip=0,netmask=0,gw=0;
	dhcp_start(&lwip_netif);//开启DHCP 
	lwipdev.dhcpstatus=0;	//正在DHCP
	printf("正在查找DHCP服务器,请稍等...........\r\n");   
	while(1)
	{ 
		printf("正在获取地址...\r\n");
		ip=lwip_netif.ip_addr.addr;		//读取新IP地址
		netmask=lwip_netif.netmask.addr;//读取子网掩码
		gw=lwip_netif.gw.addr;			//读取默认网关 
		if(ip!=0)   					//当正确读取到IP地址的时候
		{
			lwipdev.dhcpstatus=2;	//DHCP成功
 			printf("网卡en的MAC地址为:................%d.%d.%d.%d.%d.%d\r\n",lwipdev.mac[0],lwipdev.mac[1],lwipdev.mac[2],lwipdev.mac[3],lwipdev.mac[4],lwipdev.mac[5]);
			//解析出通过DHCP获取到的IP地址
			lwipdev.ip[3]=(uint8_t)(ip>>24); 
			lwipdev.ip[2]=(uint8_t)(ip>>16);
			lwipdev.ip[1]=(uint8_t)(ip>>8);
			lwipdev.ip[0]=(uint8_t)(ip);
			printf("通过DHCP获取到IP地址..............%d.%d.%d.%d\r\n",lwipdev.ip[0],lwipdev.ip[1],lwipdev.ip[2],lwipdev.ip[3]);
			//解析通过DHCP获取到的子网掩码地址
			lwipdev.netmask[3]=(uint8_t)(netmask>>24);
			lwipdev.netmask[2]=(uint8_t)(netmask>>16);
			lwipdev.netmask[1]=(uint8_t)(netmask>>8);
			lwipdev.netmask[0]=(uint8_t)(netmask);
			printf("通过DHCP获取到子网掩码............%d.%d.%d.%d\r\n",lwipdev.netmask[0],lwipdev.netmask[1],lwipdev.netmask[2],lwipdev.netmask[3]);
			//解析出通过DHCP获取到的默认网关
			lwipdev.gateway[3]=(uint8_t)(gw>>24);
			lwipdev.gateway[2]=(uint8_t)(gw>>16);
			lwipdev.gateway[1]=(uint8_t)(gw>>8);
			lwipdev.gateway[0]=(uint8_t)(gw);
			printf("通过DHCP获取到的默认网关..........%d.%d.%d.%d\r\n",lwipdev.gateway[0],lwipdev.gateway[1],lwipdev.gateway[2],lwipdev.gateway[3]);
			break;
		}else if(lwip_netif.dhcp->tries>LWIP_MAX_DHCP_TRIES) //通过DHCP服务获取IP地址失败,且超过最大尝试次数
		{  
			lwipdev.dhcpstatus=0XFF;//DHCP失败.
			//使用静态IP地址
			IP4_ADDR(&(lwip_netif.ip_addr),lwipdev.ip[0],lwipdev.ip[1],lwipdev.ip[2],lwipdev.ip[3]);
			IP4_ADDR(&(lwip_netif.netmask),lwipdev.netmask[0],lwipdev.netmask[1],lwipdev.netmask[2],lwipdev.netmask[3]);
			IP4_ADDR(&(lwip_netif.gw),lwipdev.gateway[0],lwipdev.gateway[1],lwipdev.gateway[2],lwipdev.gateway[3]);
			printf("DHCP服务超时,使用静态IP地址!\r\n");
			printf("网卡en的MAC地址为:................%d.%d.%d.%d.%d.%d\r\n",lwipdev.mac[0],lwipdev.mac[1],lwipdev.mac[2],lwipdev.mac[3],lwipdev.mac[4],lwipdev.mac[5]);
			printf("静态IP地址........................%d.%d.%d.%d\r\n",lwipdev.ip[0],lwipdev.ip[1],lwipdev.ip[2],lwipdev.ip[3]);
			printf("子网掩码..........................%d.%d.%d.%d\r\n",lwipdev.netmask[0],lwipdev.netmask[1],lwipdev.netmask[2],lwipdev.netmask[3]);
			printf("默认网关..........................%d.%d.%d.%d\r\n",lwipdev.gateway[0],lwipdev.gateway[1],lwipdev.gateway[2],lwipdev.gateway[3]);
			break;
		}  
		delay_ms(250); //延时250ms
	}
	lwip_comm_dhcp_delete();//删除DHCP任务 
}
#endif 

第三步:修改TCP/IP内核任务优先级最高

3.1 在lwipopts.h文件中修改TCP内核任务优先级为最高

#define TCPIP_THREAD_PRIO        (configMAX_PRIORITIES - 1)                //内核任务优先级

3.2 编译,没有问题,下载开发板进行验证,这里没法上传图片就不上传了

第四步:Error:..\FreeRTOS\portable\RVDS\ARM_CM4F\port.c,768错误

大家运行之后发现可以获取到动态IP,也可以建立TCP/IP的链接,只是会一直报Error:..\FreeRTOS\portable\RVDS\ARM_CM4F\port.c,768 这个错误,这是什么原因呢?

CORTEX内核的优先级是数值越小优先级越高,即0是最高优先级。FreeRTOS为了满足某些应用对中断实时性要求高的需求,使得中断优先级高于某个值之后,就不能调用操作系统的内核函数来提高实时性。这个问题是将中断的优先级设置的高于这个值,却还在中断中调用操作系统提供的API引起的。当把优先级改小也就是数值改大之后,程序能够正常运行。

下面是程序卡住的地方,从上面的注释也可以看出是因为优先级的原因。

/* The following assertion will fail if a service routine (ISR) for
			an interrupt that has been assigned a priority above
			configMAX_SYSCALL_INTERRUPT_PRIORITY calls an ISR safe FreeRTOS API
			function.  ISR safe FreeRTOS API functions must *only* be called
			from interrupts that have been assigned a priority at or below
			configMAX_SYSCALL_INTERRUPT_PRIORITY.*/
 
 
			configASSERT( ucCurrentPriority >= ucMaxSysCallPriority );

FREERTOS中的优先级数值设定的参数是configMAX_SYSCALL_INTERUPT_PRORITY。默认值是5,所以要想在中断中使用操作系统函数需要将中断的优先级设置的大于等于5.

#define configMAX_SYSCALL_INTERRUPT_PRIORITY 	 ( configLIBRARY_MAX_SYSCALL_INTERRUPT_PRIORITY << (8 - configPRIO_BITS) )

点开LAN8720.c文件,找到HAL_ETH_MspInit函数,在函数最下面2行可以看到ETH_IRQn的优先级是0,比5大,要改成6,

//ETH底层驱动,时钟使能,引脚配置
//此函数会被HAL_ETH_Init()调用
//heth:以太网句柄
void HAL_ETH_MspInit(ETH_HandleTypeDef *heth)
{
    GPIO_InitTypeDef GPIO_Initure;
    
    __HAL_RCC_ETH_CLK_ENABLE();             //开启ETH时钟
    __HAL_RCC_GPIOA_CLK_ENABLE();			//开启GPIOA时钟
	__HAL_RCC_GPIOB_CLK_ENABLE();			//开启GPIOB时钟
    __HAL_RCC_GPIOC_CLK_ENABLE();			//开启GPIOC时钟
    __HAL_RCC_GPIOG_CLK_ENABLE();			//开启GPIOG时钟
    
    /*网络引脚设置 RMII接口 
    ETH_MDIO -------------------------> PA2
    ETH_MDC --------------------------> PC1
    ETH_RMII_REF_CLK------------------> PA1
    ETH_RMII_CRS_DV ------------------> PA7
    ETH_RMII_RXD0 --------------------> PC4
    ETH_RMII_RXD1 --------------------> PC5
    ETH_RMII_TX_EN -------------------> PB11
    ETH_RMII_TXD0 --------------------> PG13
    ETH_RMII_TXD1 --------------------> PG14
    ETH_RESET-------------------------> PCF8574扩展IO*/
    
    //PA1,2,7
    GPIO_Initure.Pin=GPIO_PIN_1|GPIO_PIN_2|GPIO_PIN_7; 
    GPIO_Initure.Mode=GPIO_MODE_AF_PP;          //推挽复用
    GPIO_Initure.Pull=GPIO_NOPULL;              //不带上下拉
    GPIO_Initure.Speed=GPIO_SPEED_HIGH;         //高速
    GPIO_Initure.Alternate=GPIO_AF11_ETH;       //复用为ETH功能
    HAL_GPIO_Init(GPIOA,&GPIO_Initure);         //初始化
    
    //PB11
    GPIO_Initure.Pin=GPIO_PIN_11;               //PB11
    HAL_GPIO_Init(GPIOB,&GPIO_Initure);         //始化
    
    //PC1,4,5
    GPIO_Initure.Pin=GPIO_PIN_1|GPIO_PIN_4|GPIO_PIN_5; //PC1,4,5
    HAL_GPIO_Init(GPIOC,&GPIO_Initure);         //初始化
	
    //PG13,14
    GPIO_Initure.Pin=GPIO_PIN_13|GPIO_PIN_14;   //PG13,14
    HAL_GPIO_Init(GPIOG,&GPIO_Initure);         //初始化
    
    HAL_NVIC_SetPriority(ETH_IRQn,0,0); //此处优先级必须小于configLIBRARY_MAX_SYSCALL_INTERRUPT_PRIORITY
    HAL_NVIC_EnableIRQ(ETH_IRQn);
}

因为在ETH_IRQHandler的DMA中断接收处理函数中使用到了LWIP的协议栈,也就是使用到了freeRTOS的API,所以此处才会报错误,将HAL_NVIC_SetPriority(ETH_IRQn,0,0)改为HAL_NVIC_SetPriority(ETH_IRQn,6,0)就好了

//中断服务函数
void ETH_IRQHandler(void)
{
    while(ETH_GetRxPktSize(ETH_Handler.RxDesc))   
    {
        lwip_pkt_handle();//处理以太网数据,即将数据提交给LWIP
    }
    //清除中断标志位
    __HAL_ETH_DMA_CLEAR_IT(&ETH_Handler,ETH_DMA_IT_R); 
    __HAL_ETH_DMA_CLEAR_IT(&ETH_Handler,ETH_DMA_IT_NIS);   
}

最后,改好之后编译,就不会有错了,使用ATKKPING测试10000次,与电脑直连时ping平均速度0.05ms,还算可以

### 回答1: 这个问题中混杂了很多不同的概念。我会分别解释: 1. stm32f4xx: 这是指STMicroelectronics(意法半导体)推出的一款32位微控制器系列,它们应用广泛,有很多种型号。 2. lwIP: 这是指"lightweight IP",它是一个开源的TCP/IP协议栈。它具有轻量级、可移植、可扩展等优点,所以在一些嵌入式系统中得到应用。 3. FreeRTOS: 这是指"Free Real-time Operating System",它是一个开源的实时操作系统。它具有小巧、快速、可移植等优点,以及很多年的实战经验,所以在很多嵌入式系统中得到应用。 4. 移植: 指将特定系统的软件、驱动程序等适应不同系统,开发时多数情况下需要将软件适配到具体系统中。 综上所述, "stm32f4xx lwip freertos移植"意为适配lwIP协议栈和FreeRTOS操作系统到stm32f4xx处理器平台。 ### 回答2: STM32F4xx是属于嵌入式系统中的一种芯片,它具有高性能、低功耗、丰富外设等优点。在进行网络通信时,常常会使用到LWIP协议栈(LightWeight IP),LWIP是一个轻量级的开源TCP/IP协议栈,具有跨平台、高度可移植等特性。而FreeRTOS是一种优秀的实时操作系统,能够支持多任务、多线程、中断处理等功能,因此,将这两者进行移植,实现STM32F4xx芯片的网络通信,特别是TCP/IP通信,将是一个很有挑战的任务。 移植LWIP协议栈需要进行以下步骤: 第一步:根据STM32F4xx芯片的电路板、内存大小、外设特性等环境,进行LWIP协议栈的移植。主要包括系统初始化、网络驱动、网络协议栈、网络接口等方面的内容。 第步:通过LWIP的API接口,实现协议的配置,包括IP地址、掩码、网关、DNS服务器等,并根据不同的协议类型(如TCP、UDP、ICMP等)进行配置。 第三步:通过FreeRTOS的API接口,将网络协议栈与操作系统进行整合,实现多任务并发处理、中断处理、定时器计数等功能,并保证系统稳定性和实时性。 在进行STM32F4xx lwip freertos移植时,需要注意以下几个方面: 一、内存管理:STM32F4xx芯片的RAM和Flash比较小,需要合理地分配内存,避免资源浪费和系统崩溃。 、时钟配置:LWIPFreeRTOS都需要使用操作系统的时钟进行计数和同步。因此,需要把选定的操作系统时钟映射到芯片上的RC/XTAL或内部晶振,保证时钟精度和稳定性。 三、中断处理:由于网络协议栈需要进行中断处理,因此需要注意中断的优先级和中断处理函数的编写。中断处理函数需要精简、高效,不影响系统的实时性和稳定性。 四、网络接口:STM32F4xx芯片支持多种不同的网络接口,在移植中需要根据具体的需求,选择相应的硬件接口进行配置。并根据不同的接口类型,进行网络驱动程序的编写和配置。 总之,STM32F4xx lwip freertos移植并不是一项简单的任务,需要开发者具备深厚的嵌入式开发经验和相关技能。通过合理的规划、持续的优化,可以实现高效稳定的网络通信。 ### 回答3: 随着物联网应用的日益普及,嵌入式系统中使用lwIPFreeRTOS的需求越来越大。STM32F4系列是一种高性能的嵌入式微控制器,它支持lwIPFreeRTOS,因此很适合用于物联网领域的开发。这篇文章将介绍STM32F4xx lwipFreeRTOS移植步骤。 1. 系统架构 在移植前,需要先了解STM32F4xx系列的架构。STM32F4xx系列的主要子系统有: - Cortex-M4内核 - 系统存储器(SRAM)和闪存 - 外设:USART、SPI、I2C、USB、以太网等 - DMA控制器 lwIP是一个轻量级的IP协议组件库,它能够在嵌入式系统中实现TCP/IP协议栈。在STM32F4xx系列中,lwIPFreeRTOS可以运行在主内存中。为了获得更好的性能,建议采用SRAM作为系统存储器,并为lwIPFreeRTOS预留足够的内存空间。 2. 移植步骤 2.1. 配置IDE开发环境 移植lwIPFreeRTOS需要用到IDE工具,比如Keil、IAR和TrueStudio等。在开发过程中,需要配置好编译器、调试器和开发板等相关环境。 2.2. 配置FreeRTOS FreeRTOSSTM32F4xx系列中的线程操作提供支持。在移植过程中,需要设置线程的优先级、任务管理器、内存管理器和时间管理器等。同时,还需要对FreeRTOS进行适当的调优,以获得更好的性能和可靠性。 2.3. 配置lwIP lwIP移植涉及到网络协议栈,需要对其进行详细的配置。首先要配置网络接口,包括MAC和IP地址、子网掩码、网关和DNS服务器等。然后需要配置协议栈参数,包括缓冲区的大小、超时时间和最大传输单元(MTU)等。最后还需要配置协议栈服务,包括DHCP、NAT、HTTP、FTP和SMTP等。 2.4. 配置硬件平台 在移植过程中,需要配置硬件平台,包括外设控制器、DMA控制器和引脚映射等。在使用网卡时,还需要配置PHY芯片。 3. 移植测试 在完成lwIPFreeRTOS移植后,需要进行测试以确保其功能正常。测试方法包括: - 使用ping测试网络连接 - 使用telnet实现远程命令 - 使用HTTP服务器进行数据交互 移植过程中可能会遇到各种问题,比如芯片引脚分配不当、外设驱动程序错误、协议栈配置不正确等。为了快速诊断问题,可以使用调试工具(比如JTAG调试器)或日志文件进行调试。 总之,STM32F4xx lwipFreeRTOS移植并不是一件容易的事,它需要开发人员具备扎实的嵌入式系统和网络编程技能。当然,一旦成功地移植了它们,就可以让设备更好地应用于物联网领域,为用户提供更为方便、高效和安全的服务。
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值