原创 一、FreeRTOS移植-stm32

2020-8-18 17:53 3069 16 3 分类: 汽车电子 文集: FreeRTOS

FreeRTOS简介

1 RTOS运行过程:

RTOS系统的运行如图所示,首先会划分很多个任务,每个任务都具有任务优先级,高优先级的任务会先运行。但是,中断服务函数(如 TIM、USART)运行的优先级是最高的,也就是说,中断能打断其他所有的任务。对于RTOS而言,运行的永远是处于就绪态的最高优先级的任务。

 

2 FreeRTOS简介

FreeRTOS官网:http://www.freertos.org

FreeRTOS源码内容如下所示

 

FreeRTOS移植

1 移植过程,准备一个简单的工程,以流水灯为例,原始工程结构如下:

2 新建 FreeRTOS 文件夹,并复制全部 FreeRTOS 源码;

 portable 文件夹中,我们只需要留下 keil、MemMang 和 RVDS这三个文件夹,其他的都可以删除掉,如下所示:

3 打开原始工程,新建两个分组 FreeRTOS_CORE(6个内核相关的文件) 和 FreeRTOS_PORTABLE(2个和移植相关的文件),然后向这两个分组添加文件:

添加结果如图所示:

4 添加2个头文件路径(include 和 RVDS/ARM_CM3):

5 编译工程,提示找不到 FreeRTOSConfig.h 文件

这个文件是 FreeRTOS 的配置文件,不属于源码中的一部分,用户可以根据自己的需要来进行配置,用于裁剪 FreeRTOS,FreeRTOSConfig.h 在 demo 文件夹中可以找到:

我们直接使用野火提供的 FreeRTOSConfig.h 配置文件,复制到 include 文件夹下;

再重新编译,结果如下(提示这两个函数重复定义):

直接在stm32f10x_it.c 中注释掉这两个函数:

然后,还需要在 stm32f10x_it.c 中 Systick_Handler() 中断服务函数中,调用 FreeRTOS 的系统节拍处理函数;

这样,FreeRTOS移植完成。

 

测试例程:

  1. #include "stm32f10x.h"
  2. #include "bsp_led.h"
  3. #include "bsp_usart.h"
  4. #include "FreeRTOS.h"
  5. #include "task.h"
  6. // 开始任务句柄、任务栈大小、优先级和任务主体函数
  7. TaskHandle_t Start_Task_Handler;
  8. #define START_STK_SIZE 256
  9. #define START_TASK_PRIO 1
  10. void start_task(void *pvParameters);
  11. // 任务1句柄、任务栈大小、优先级和任务主体函数
  12. TaskHandle_t Task1_Handler;
  13. #define TASK1_STK_SIZE 256
  14. #define TASK1_TASK_PRIO 2
  15. void task1_task(void *pvParameters);
  16. // 任务2句柄、任务栈大小、优先级和任务主体函数
  17. TaskHandle_t Task2_Handler;
  18. #define TASK2_STK_SIZE 256
  19. #define TASK2_TASK_PRIO 3
  20. void task2_task(void *pvParameters);
  21. int main(void)
  22. {
  23. LED_GPIO_Config();
  24. USART_config();
  25. // 创建开始任务,然后在开始任务中创建其他任务,再删除开始任务
  26. xTaskCreate((TaskFunction_t)start_task,
  27. (const char *)"start_task",
  28. (uint16_t)START_STK_SIZE,
  29. (void *)NULL,
  30. (UBaseType_t)START_TASK_PRIO,
  31. (TaskHandle_t *)&Start_Task_Handler);
  32. vTaskStartScheduler(); // 开始任务调度器
  33. }
  34. void start_task(void *pvParameters)
  35. {
  36. taskENTER_CRITICAL(); // 进入临界区
  37. // 创建两个任务
  38. xTaskCreate((TaskFunction_t)task1_task,
  39. (const char *)"task1_task",
  40. (uint16_t)TASK1_STK_SIZE,
  41. (void *)NULL,
  42. (UBaseType_t)TASK1_TASK_PRIO,
  43. (TaskHandle_t *)&Task1_Handler);
  44. xTaskCreate((TaskFunction_t)task2_task,
  45. (const char *)"task2_task",
  46. (uint16_t)TASK2_STK_SIZE,
  47. (void *)NULL,
  48. (UBaseType_t)TASK2_TASK_PRIO,
  49. (TaskHandle_t *)&Task2_Handler);
  50. vTaskDelete(Start_Task_Handler); // 删除开始任务
  51. taskEXIT_CRITICAL(); // 退出临界区
  52. }
  53. // 任务1:翻转led
  54. void task1_task(void *pvParameters)
  55. {
  56. uint8_t flag = 0;
  57. for (; ;)
  58. {
  59. flag = !flag;
  60. if (flag)
  61. {
  62. LED_RED;
  63. }
  64. else
  65. {
  66. LED_ALL_OFF;
  67. }
  68. vTaskDelay(500);
  69. }
  70. }
  71. // 任务2:串口打印
  72. void task2_task(void *pvParameters)
  73. {
  74. uint16_t num = 0;
  75. for (; ;)
  76. {
  77. printf("task2 now running %d times.\n", num++);
  78. vTaskDelay(500);
  79. }
  80. }

运行结果如图所示:

 转载于:https://blog.csdn.net/dingyc_ee/article/details/103929293

版权归原作者所有。

PARTNER CONTENT

文章评论0条评论)

登录后参与讨论
EE直播间
更多
我要评论
0
16
关闭 站长推荐上一条 /3 下一条