FreeRTOS(动态内存管理)

资料来源于硬件家园:资料汇总 - FreeRTOS实时操作系统课程(多任务管理)

目录

一、动态内存管理介绍

1、heap_1

2、heap_2

3、heap_3

4、heap_4

5、heap_5

二、动态内存总结与应用

1、heap_1

2、heap_4

3、heap_5

三、内存管理编程测试

1、heap_4

2、heap_1

3、heap_5

一、动态内存管理介绍

动态内存管理是 FreeRTOS 非常重要的一项功能,前面章节讲解的任务创建、信号量、消息队列、事件标志组、互斥信号量、软件定时器组等需要的 RAM 空间都是通过动态内存管理从 FreeRTOSConfig.h 文件定义的 heap 空间中申请的。

FreeRTOS 支持 5 种动态内存管理方案,分别通过文件 heap_1,heap_2,heap_3,heap_4 和 heap_5实现。

1、heap_1

heap_1 动态内存管理方式是五种动态内存管理方式中最简单的,这种方式的动态内存管理一旦申请了相应内存后,是不允许被释放的。

尽管如此,这种方式的动态内存管理还是满足大部分嵌入式应用的,因为这种嵌入式应用在系统启动阶段就完成了任务创建、事件标志组、信号量、消息队列等资源的创建,而且这些资源是整个嵌入式应用过程中一直要使用的,所以也就不需要删除,不需要释放内存。

2、heap_2

与 heap_1 动态内存管理方式不同,heap_2 动态内存管理利用了最适应算法,并且支持内存释放。但是 heap_2 不支持内存碎片整理,动态内存管理方式heap_4 支持内存碎片整理。

3、heap_3

这种方式实现的动态内存管理是对编译器提供的 malloc 和 free 函数进行了封装,保证是线程安全的。

4、heap_4

与 heap_2 动态内存管理方式不同,heap_4 动态内存管理利用了最适应算法,且支持内存碎片的回收并将其整理为一个大的内存块。

5、heap_5

有时候我们希望 FreeRTOSConfig.h 文件中定义的 heap 空间可以采用不连续的内存区,比如我们希望可以将其定义在内部 SRAM 一部分,外部 SRAM 一部分,此时我们就可以采用 heap_5 动态内存管理方式。另外,heap_5 动态内存管理是在 heap_4 的基础上实现的。

二、动态内存总结与应用

五种动态内存管理方式简单总结如下,实际项目中,用户根据需要选择合适的:

heap_1:五种方式里面最简单的,但是申请的内存不允许释放

heap_2:支持动态内存的申请和释放,但是不支持内存碎片的处理

heap_3:将编译器自带的 malloc 和 free 函数进行简单的封装

heap_4:支持动态内存的申请和释放,支持内存碎片处理

heap_5:在 heap_4 的基础上支持将动态内存设置在不连续的区域上

1、heap_1

FreeRTOS 的动态内存大小在 FreeRTOSConfig.h 文件中进行了定义:

#define configTOTAL_HEAP_SIZE  ( ( size_t ) ( 17 * 1024 ) ) //单位字节

用户通过函数 xPortGetFreeHeapSize 就能获得 FreeRTOS 动态内存的剩余,进而可以根据剩余情况优化动态内存的大小。

heap_1 方式的动态内存管理有以下特点:

① 项目应用不需要删除任务、信号量、消息队列等已经创建的资源。

② 具有时间确定性,即申请动态内存的时间是固定的并且不会产生内存碎片。

③ 确切的说这是一种静态内存分配,因为申请的内存是不允许被释放掉的。

2、heap_4

FreeRTOS 的动态内存大小在 FreeRTOSConfig.h 文件中进行了定义:

#define configTOTAL_HEAP_SIZE  ( ( size_t ) ( 17 * 1024 ) ) //单位字节

用户通过函数 xPortGetFreeHeapSize 就能获得 FreeRTOS 动态内存的剩余,但是不提供动态内存是如何被分配成各个小内存块的信息。使用函数 xPortGetMinimumEverFreeHeapSize 能够获取从系统启动到当前时刻的动态内存最小剩余,从而用户就可以根据剩余情况优化动态内存的大小。

heap_4 方式的动态内存管理有以下特点:

① 可以用于需要重复的创建和删任务、信号量、事件标志组、软件定时器等内部资源的场合。

② 随机的调用 pvPortMalloc() 和 vPortFree(),且每次申请的大小都不同,也不会像 heap_2 那样产生很多的内存碎片。

③ 不具有时间确定性,即申请动态内存的时间不是确定的

3、heap_5

heap_5 动态内存管理是通过函数 vPortDefineHeapRegions 进行初始化的,也就是说用户在创建任务 FreeRTOS 的内部资源前要优先级调用这个函数 vPortDefineHeapRegions,否则是无法通过函数pvPortMalloc 申请到动态内存的。

函数 vPortDefineHeapRegions 定义不同段的内存空间采用了下面这种结构体:

定义的时候要注意两个问题,一个是内存段结束时要定义 NULL。另一个是内存段的地址是从低地址到高地址排列。

用户通过函数 xPortGetFreeHeapSize 就能获得 FreeRTOS 动态内存的剩余,但是不提供动态内存是如何被分配成各个小内存块的信息。使用函数 xPortGetMinimumEverFreeHeapSize 能够获取从系统启动到当前时刻的动态内存最小剩余,从而用户就可以根据剩余情况优化动态内存的大小。

三、内存管理编程测试

分别测试heap_4heap_1heap_5

1、heap_4

	  if(KeyCode==KEY0)
	  {
		sprintf(buff,"当前剩余动态内存大小 %u Bytes\r\n",xPortGetFreeHeapSize());
		HAL_UART_Transmit(&huart2, (uint8_t*)buff,strlen(buff), HAL_MAX_DELAY);

		sprintf(buff,"系统启动至当前时刻的动态内存最小剩余 %u Bytes\r\n",xPortGetMinimumEverFreeHeapSize());
		HAL_UART_Transmit(&huart2, (uint8_t*)buff,strlen(buff), HAL_MAX_DELAY);
	  }

输出内存信息:

2、heap_1

	  if(KeyCode==KEY0)
	  {
		sprintf(buff,"当前剩余动态内存大小 %u Bytes\r\n",xPortGetFreeHeapSize());
		HAL_UART_Transmit(&huart2, (uint8_t*)buff,strlen(buff), HAL_MAX_DELAY);

		//sprintf(buff,"系统启动至当前时刻的动态内存最小剩余 %u Bytes\r\n",xPortGetMinimumEverFreeHeapSize());
		//HAL_UART_Transmit(&huart2, (uint8_t*)buff,strlen(buff), HAL_MAX_DELAY);
	  }

 

如果调用vPortFree()函数释放内存,系统会死机,heap_1内存管理方案不支持内存释放,导致死机,

3、heap_5

定义内存区域:

  /* USER CODE BEGIN Init */
	HeapRegion_t xHeapRegions[]=
	{
		{(uint8_t*)0x2000B000UL,0x1400},//5K
		{(uint8_t*)0x2000D800UL,0x2800},//10K
		{NULL,0}
	};
	vPortDefineHeapRegions(xHeapRegions);
  /* USER CODE END Init */
	  if(KeyCode==KEY0)
	  {
		sprintf(buff,"当前剩余动态内存大小 %u Bytes\r\n",xPortGetFreeHeapSize());
		HAL_UART_Transmit(&huart2, (uint8_t*)buff,strlen(buff), HAL_MAX_DELAY);

		sprintf(buff,"系统启动至当前时刻的动态内存最小剩余 %u Bytes\r\n",xPortGetMinimumEverFreeHeapSize());
		HAL_UART_Transmit(&huart2, (uint8_t*)buff,strlen(buff), HAL_MAX_DELAY);
	  }

输出内存信息:

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
FreeRTOS 中进行动态内存申请,可以使用 FreeRTOS 内置的内存管理函数。FreeRTOS 提供了两种内存管理方案:heap_1 和 heap_5。 1. heap_1:heap_1 是最简单的内存管理方案,它使用 C 标准库中的 malloc() 和 free() 函数来实现内存的分配和释放。你可以通过在 FreeRTOSConfig.h 文件中定义 configUSE_MALLOC 宏来启用 heap_1 内存管理方案。 2. heap_5:heap_5 是更高级的内存管理方案,它使用 FreeRTOS 内置的内存管理函数 pvPortMalloc() 和 vPortFree() 来实现内存的分配和释放。heap_5 允许你通过定义 configAPPLICATION_ALLOCATED_HEAP 宏来指定一个连续的内存区域供 FreeRTOS 作为堆内存使用。 以下是在 FreeRTOS 中进行动态内存申请的示例代码: ```c #include "FreeRTOS.h" #include "task.h" void vTaskFunction(void *pvParameters) { // 动态分配内存 int *ptr = (int *)pvPortMalloc(sizeof(int)); if (ptr != NULL) { // 内存分配成功,可以进行操作 *ptr = 123; // ... // 释放内存 vPortFree(ptr); } vTaskDelete(NULL); } int main(void) { // 创建任务 xTaskCreate(vTaskFunction, "Task", configMINIMAL_STACK_SIZE, NULL, tskIDLE_PRIORITY, NULL); // 启动调度器 vTaskStartScheduler(); return 0; } ``` 在上述示例中,任务函数 vTaskFunction() 中使用 pvPortMalloc() 函数动态分配了一个 int 类型的内存,并在使用完后通过 vPortFree() 函数释放内存。请注意,内存的释放要确保在不再使用时执行,以避免内存泄漏。 需要注意的是,动态内存分配是一项复杂的任务,可能涉及到内存碎片、多任务竞争等问题。因此,在使用动态内存时,需谨慎规划和管理,以确保系统的可靠性和稳定性。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Zhang丶&|!

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值