存管理实验使用 heap_4.c 方案进行内存管理测试, 创建了两个任务,分别是 LED 任务与内存管理测试任务,内存管理测试任务通过检测按键是否按下来申请内存或释放内存,当申请内存成功就像该内存写入一些数据,如当前系统的时间等信息,并且通过串口输出相关信息; LED 任务是将 LED 翻转,表示系统处于运行状态。在不需要再使用内存时,注意要及时释放该段内存,避免内存泄露,

    1. /*
    2. *************************************************************************
    3. * 包含的头文件
    4. *************************************************************************
    5. */
    6. /* FreeRTOS头文件 */
    7. #include "FreeRTOS.h"
    8. #include "task.h"
    9. /* 开发板硬件bsp头文件 */
    10. #include "bsp_led.h"
    11. #include "bsp_usart.h"
    12. #include "bsp_key.h"
    13. /**************************** 任务句柄 ********************************/
    14. /*
    15. * 任务句柄是一个指针,用于指向一个任务,当任务创建好之后,它就具有了一个任务句柄
    16. * 以后我们要想操作这个任务都需要通过这个任务句柄,如果是自身的任务操作自己,那么
    17. * 这个句柄可以为NULL。
    18. */
    19. static TaskHandle_t AppTaskCreate_Handle = NULL;/* 创建任务句柄 */
    20. static TaskHandle_t LED_Task_Handle = NULL;/* LED_Task任务句柄 */
    21. static TaskHandle_t Test_Task_Handle = NULL;/* Test_Task任务句柄 */
    22. /******************************* 全局变量声明 ************************************/
    23. /*
    24. * 当我们在写应用程序的时候,可能需要用到一些全局变量。
    25. */
    26. uint8_t *Test_Ptr = NULL;
    27. /*
    28. *************************************************************************
    29. * 函数声明
    30. *************************************************************************
    31. */
    32. static void AppTaskCreate(void);/* 用于创建任务 */
    33. static void LED_Task(void* pvParameters);/* LED_Task任务实现 */
    34. static void Test_Task(void* pvParameters);/* Test_Task任务实现 */
    35. static void BSP_Init(void);/* 用于初始化板载相关资源 */
    36. /*****************************************************************
    37. * @brief 主函数
    38. * @param 无
    39. * @retval 无
    40. * @note 第一步:开发板硬件初始化
    41. 第二步:创建APP应用任务
    42. 第三步:启动FreeRTOS,开始多任务调度
    43. ****************************************************************/
    44. int main(void)
    45. {
    46. BaseType_t xReturn = pdPASS;/* 定义一个创建信息返回值,默认为pdPASS */
    47. /* 开发板硬件初始化 */
    48. BSP_Init();
    49. printf("这是一个[野火]-STM32全系列开发板-FreeRTOS内存管理实验\n");
    50. printf("按下KEY1申请内存,按下KEY2释放内存\n");
    51. /* 创建AppTaskCreate任务 */
    52. xReturn = xTaskCreate((TaskFunction_t )AppTaskCreate, /* 任务入口函数 */
    53. (const char* )"AppTaskCreate",/* 任务名字 */
    54. (uint16_t )512, /* 任务栈大小 */
    55. (void* )NULL,/* 任务入口函数参数 */
    56. (UBaseType_t )1, /* 任务的优先级 */
    57. (TaskHandle_t* )&AppTaskCreate_Handle);/* 任务控制块指针 */
    58. /* 启动任务调度 */
    59. if(pdPASS == xReturn)
    60. vTaskStartScheduler(); /* 启动任务,开启调度 */
    61. else
    62. return -1;
    63. while(1); /* 正常不会执行到这里 */
    64. }
    65. /***********************************************************************
    66. * @ 函数名 : AppTaskCreate
    67. * @ 功能说明: 为了方便管理,所有的任务创建函数都放在这个函数里面
    68. * @ 参数 : 无
    69. * @ 返回值 : 无
    70. **********************************************************************/
    71. static void AppTaskCreate(void)
    72. {
    73. BaseType_t xReturn = pdPASS;/* 定义一个创建信息返回值,默认为pdPASS */
    74. taskENTER_CRITICAL(); //进入临界区
    75. /* 创建LED_Task任务 */
    76. xReturn = xTaskCreate((TaskFunction_t )LED_Task, /* 任务入口函数 */
    77. (const char* )"LED_Task",/* 任务名字 */
    78. (uint16_t )512, /* 任务栈大小 */
    79. (void* )NULL, /* 任务入口函数参数 */
    80. (UBaseType_t )2, /* 任务的优先级 */
    81. (TaskHandle_t* )&LED_Task_Handle);/* 任务控制块指针 */
    82. if(pdPASS == xReturn)
    83. printf("创建LED_Task任务成功\n");
    84. /* 创建Test_Task任务 */
    85. xReturn = xTaskCreate((TaskFunction_t )Test_Task, /* 任务入口函数 */
    86. (const char* )"Test_Task",/* 任务名字 */
    87. (uint16_t )512, /* 任务栈大小 */
    88. (void* )NULL,/* 任务入口函数参数 */
    89. (UBaseType_t )3, /* 任务的优先级 */
    90. (TaskHandle_t* )&Test_Task_Handle);/* 任务控制块指针 */
    91. if(pdPASS == xReturn)
    92. printf("创建Test_Task任务成功\n\n");
    93. vTaskDelete(AppTaskCreate_Handle); //删除AppTaskCreate任务
    94. taskEXIT_CRITICAL(); //退出临界区
    95. }
    96. /**********************************************************************
    97. * @ 函数名 : LED_Task
    98. * @ 功能说明: LED_Task任务主体
    99. * @ 参数 :
    100. * @ 返回值 : 无
    101. ********************************************************************/
    102. static void LED_Task(void* parameter)
    103. {
    104. while (1)
    105. {
    106. LED1_TOGGLE;
    107. vTaskDelay(1000);/* 延时1000个tick */
    108. }
    109. }
    110. /**********************************************************************
    111. * @ 函数名 : Test_Task
    112. * @ 功能说明: Test_Task任务主体
    113. * @ 参数 :
    114. * @ 返回值 : 无
    115. ********************************************************************/
    116. static void Test_Task(void* parameter)
    117. {
    118. uint32_t g_memsize;
    119. while (1)
    120. {
    121. if( Key_Scan(KEY1_GPIO_PORT,KEY1_GPIO_PIN) == KEY_ON )
    122. {
    123. /* KEY1 被按下 */
    124. if(NULL == Test_Ptr)
    125. {
    126. /* 获取当前内存大小 */
    127. g_memsize = xPortGetFreeHeapSize();
    128. printf("系统当前内存大小为 %d 字节,开始申请内存\n",g_memsize);
    129. Test_Ptr = pvPortMalloc(1024);
    130. if(NULL != Test_Ptr)
    131. {
    132. printf("内存申请成功\n");
    133. printf("申请到的内存地址为%#x\n",(int)Test_Ptr);
    134. /* 获取当前内剩余存大小 */
    135. g_memsize = xPortGetFreeHeapSize();
    136. printf("系统当前内存剩余存大小为 %d 字节\n",g_memsize);
    137. //向Test_Ptr中写入当数据:当前系统时间
    138. sprintf((char*)Test_Ptr,"当前系统TickCount = %d \n",xTaskGetTickCount());
    139. printf("写入的数据是 %s \n",(char*)Test_Ptr);
    140. }
    141. }
    142. else
    143. {
    144. printf("请先按下KEY2释放内存再申请\n");
    145. }
    146. }
    147. if( Key_Scan(KEY2_GPIO_PORT,KEY2_GPIO_PIN) == KEY_ON )
    148. {
    149. /* KEY2 被按下 */
    150. if(NULL != Test_Ptr)
    151. {
    152. printf("释放内存\n");
    153. vPortFree(Test_Ptr); //释放内存
    154. Test_Ptr=NULL;
    155. /* 获取当前内剩余存大小 */
    156. g_memsize = xPortGetFreeHeapSize();
    157. printf("系统当前内存大小为 %d 字节,内存释放完成\n",g_memsize);
    158. }
    159. else
    160. {
    161. printf("请先按下KEY1申请内存再释放\n");
    162. }
    163. }
    164. vTaskDelay(20);/* 延时20个tick */
    165. }
    166. }
    167. /***********************************************************************
    168. * @ 函数名 : BSP_Init
    169. * @ 功能说明: 板级外设初始化,所有板子上的初始化均可放在这个函数里面
    170. * @ 参数 :
    171. * @ 返回值 : 无
    172. *********************************************************************/
    173. static void BSP_Init(void)
    174. {
    175. /*
    176. * STM32中断优先级分组为4,即4bit都用来表示抢占优先级,范围为:0~15
    177. * 优先级分组只需要分组一次即可,以后如果有其他的任务需要用到中断,
    178. * 都统一用这个优先级分组,千万不要再分组,切忌。
    179. */
    180. NVIC_PriorityGroupConfig( NVIC_PriorityGroup_4 );
    181. /* LED 初始化 */
    182. LED_GPIO_Config();
    183. /* 串口初始化 */
    184. USART_Config();
    185. /* 按键初始化 */
    186. Key_GPIO_Config();
    187. }
    188. /********************************END OF FILE****************************/