> 文档中心 > GD32F310移植FreeRTOS

GD32F310移植FreeRTOS


【GD32F310开发板试用】GD32F310移植FreeRTOS

项目地址:https://github.com/kings669/GD32F310-FreeRTOS

环境说明

首先我先说明我的环境:
我使用keil5的版本是V5.36,电脑系统是WIN10,使用的固件库版本是从官网下载的最新版2.2.0(官网下载地址:http://www.gd32mcu.com/cn/download/7?kw=GD32F3),我所移植的FreeRTOS版本为:v202112.00(官网下载地址:https://www.freertos.org/a00104.html)
GD32F310移植FreeRTOS

开发板信息

我们在来看看关于这款开发板:
GD32F310K-START 评估板使用 GD32F310K8T6 作为主控制器。LED的引脚是PA8,之后我们会用到。
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-FDC7tB6e-1648836339477)(/img/bVbtN7)]

我们在看看Flash和RAM大小分别是64KB和8KB。到时候我们会在FreeRTOSConfig.h里面设置大小。
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-4k9ZtLFw-1648836339478)(/img/bVbtN9)]板载的调试器是GDLink

移植过程

可以参考我项目里面的资料,需要的文件均放在了doc文件夹中。
建好文件夹,命好名字:
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-jjTaljZq-1648836339479)(/img/bVbtTg)]

我们把FreeRTOS的文件放在中间件Middleware文件夹中。
下载好的FreeRTOS文件夹:
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-fjadvaar-1648836339479)(/img/bVbtTh)]

我们选用FreeRTOS就行,没必要使用Plus,我们需要的文件夹都在FreeRTOS中,将它复制到自己的项目文件中。
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-QZHSw5XX-1648836339480)(/img/bVbtTi)]

我们把下载好的固件库以及CMSIS文件放在我们建好的Drivers文件夹中。
GD32F310移植FreeRTOS

接下来就keil5相关的设置:
1.预定义:记得勾选支持C99,以及添加头文件路径。
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-hTziEnH9-1648836339481)(/img/bVbtTm)]

2.keil5中文件添加,按照文件夹的来做
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-DWBlLhMN-1648836339482)(/img/bVbtTo)]

这里要稍微注意的是:port.c是选择图中路径的文件(由于我们芯片是M4的内核),内存管理我们选择heap_4.c
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-DLIiqQ1Z-1648836339483)(/img/bVbtTp)]

GD32F310移植FreeRTOS

到这里我们还缺少一个FreeRTOS的配置文件,FreeRTOSConfig.h;
内容是:

#ifndef FREERTOS_CONFIG_H#define FREERTOS_CONFIG_H#include "main.h"/*----------------------------------------------------------- * Application specific definitions. * * These definitions should be adjusted for your particular hardware and * application requirements. * * THESE PARAMETERS ARE DESCRIBED WITHIN THE 'CONFIGURATION' SECTION OF THE * FreeRTOS API DOCUMENTATION AVAILABLE ON THE FreeRTOS.org WEB SITE. * * See http://www.freertos.org/a00110.html. *----------------------------------------------------------*//* Ensure stdint is only used by the compiler, and not the assembler. */#if defined(__ICCARM__)   ||  defined(__CC_ARM) ||  defined(__GNUC__)#include extern uint32_t SystemCoreClock;#endif#define configUSE_PREEMPTION     1   //1使用抢占式内核,0使用协程#define configUSE_TIME_SLICING   1   //1使能时间片调度(默认式使能的)  #define configUSE_PORT_OPTIMISED_TASK_SELECTION1    //1启用特殊方法来选择下一个要运行的任务    //一般是硬件计算前导零指令,如果所使用的    //MCU没有这些硬件指令的话此宏应该设置为0!#define configUSE_TICKLESS_IDLE 0    //1启用低功耗tickless模式#define configUSE_QUEUE_SETS 1    //为1时启用队列#define configUSE_IDLE_HOOK0    //1,使用空闲钩子;0,不使用#define configUSE_TICK_HOOK0    //1,使用时间片钩子;0,不使用#define configCPU_CLOCK_HZ( SystemCoreClock ) //CPU频率#define configTICK_RATE_HZ( ( TickType_t ) 1000 ) //时钟节拍频率,这里设置为1000,周期就是1ms#define configMAX_PRIORITIES( 16 )  //可使用的最大优先级#define configMINIMAL_STACK_SIZE( ( unsigned short ) 128 )  //空闲任务使用的堆栈大小#define configTOTAL_HEAP_SIZE( ( size_t ) ( 6 * 1024 ) )   //系统所有总的堆大小#define configMAX_TASK_NAME_LEN( 16 )  //任务名字字符串长度#define configUSE_16_BIT_TICKS0   //系统节拍计数器变量数据类型,  //1表示为16位无符号整形,0表示为32位无符号整形#define INCLUDE_xTaskGetHandle 1#define configIDLE_SHOULD_YIELD1   //为1时空闲任务放弃CPU使用权给其他同优先级的用户任务#define configUSE_TASK_NOTIFICATIONS    1   //为1时开启任务通知功能,默认开启#define configUSE_MUTEXES1   //为1时使用互斥信号量#define configQUEUE_REGISTRY_SIZE8   //不为0时表示启用队列记录,具体的值是可以  //记录的队列和信号量最大数目。#define configCHECK_FOR_STACK_OVERFLOW0   //大于0时启用堆栈溢出检测功能,如果使用此功能  //用户必须提供一个栈溢出钩子函数,如果使用的话#define configUSE_RECURSIVE_MUTEXES1   //为1时使用递归互斥信号量#define configUSE_MALLOC_FAILED_HOOK0   //1使用内存申请失败钩子函数#define configUSE_APPLICATION_TASK_TAG0#define configUSE_COUNTING_SEMAPHORES1   //为1时使用计数信号量#define configSUPPORT_DYNAMIC_ALLOCATION 1  //支持动态内存申请#define configUSE_TRACE_FACILITY1   //为1启用可视化跟踪调试#define configGENERATE_RUN_TIME_STATS0   //为1时启用运行时间统计功能#define configUSE_STATS_FORMATTING_FUNCTIONS1//与宏configUSE_TRACE_FACILITY同时为1时会编译下面3个函数//prvWriteNameToBuffer(),vTaskList(),//vTaskGetRunTimeStats()#define INCLUDE_uxTaskGetStackHighWaterMark 1/* Co-routine definitions. */#define configUSE_CO_ROUTINES 0     //为1时启用协程,启用协程以后必须添加文件croutine.c#define configMAX_CO_ROUTINE_PRIORITIES ( 2 )   //协程的有效优先级数目/* Software timer definitions. */#define configUSE_TIMERS1 //为1时启用软件定时器#define configTIMER_TASK_PRIORITY( 2 )   //软件定时器优先级#define configTIMER_QUEUE_LENGTH10      //软件定时器队列长度#define configTIMER_TASK_STACK_DEPTH( configMINIMAL_STACK_SIZE * 2 )    //软件定时器任务堆栈大小/* Set the following definitions to 1 to include the API function, or zeroto exclude the API function. */#define INCLUDE_vTaskPrioritySet1#define INCLUDE_uxTaskPriorityGet1#define INCLUDE_vTaskDelete1#define INCLUDE_vTaskCleanUpResources1#define INCLUDE_vTaskSuspend1#define INCLUDE_vTaskDelayUntil1#define INCLUDE_vTaskDelay1/* Cortex-M specific definitions. */#ifdef __NVIC_PRIO_BITS/* __BVIC_PRIO_BITS will be specified when CMSIS is being used. */#define configPRIO_BITS__NVIC_PRIO_BITS#else#define configPRIO_BITS4 /* 15 priority levels */#endif/* The lowest interrupt priority that can be used in a call to a "set priority"function. */#define configLIBRARY_LOWEST_INTERRUPT_PRIORITY0xf     //中断最低优先级/* The highest interrupt priority that can be used by any interrupt serviceroutine that makes calls to interrupt safe FreeRTOS API functions.  DO NOT CALLINTERRUPT SAFE FREERTOS API FUNCTIONS FROM ANY INTERRUPT THAT HAS A HIGHERPRIORITY THAN THIS! (higher priorities are lower numeric values. */#define configLIBRARY_MAX_SYSCALL_INTERRUPT_PRIORITY5//系统可管理的最高中断优先级/* Interrupt priorities used by the kernel port layer itself.  These are genericto all Cortex-M ports, and do not rely on any particular library functions. */#define configKERNEL_INTERRUPT_PRIORITY ( configLIBRARY_LOWEST_INTERRUPT_PRIORITY << (8 - configPRIO_BITS) )/* !!!! configMAX_SYSCALL_INTERRUPT_PRIORITY must not be set to zero !!!!See http://www.FreeRTOS.org/RTOS-Cortex-M3-M4.html. */#define configMAX_SYSCALL_INTERRUPT_PRIORITY ( configLIBRARY_MAX_SYSCALL_INTERRUPT_PRIORITY << (8 - configPRIO_BITS) )/* Normal assert() semantics without relying on the provision of an assert.hheader file. */#define configASSERT( x ) if( ( x ) == 0 ) { taskDISABLE_INTERRUPTS(); for( ;; ); }/* Definitions that map the FreeRTOS port interrupt handlers to their CMSISstandard names. */#define vPortSVCHandler SVC_Handler#define xPortPendSVHandler PendSV_Handler//#define xPortSysTickHandler SysTick_Handler#endif /* FREERTOS_CONFIG_H */

点灯测试

看了一下官方的例程,还是有点不太习惯,我习惯是结构体的写法。所以我把此处进行了改写,并且打算将后面的外设也一一写完,感兴趣的可以关注项目。
灯的引脚是PA8
在这里插入图片描述

main.c文件

#include "gd32f3x0.h"#include "systick.h"#include "FreeRTOS.h"#include "task.h"#include "gpio.h"#define START_TASK_PRIO1#define START_STK_SIZE 128  TaskHandle_t StartTask_Handler;void start_task(void *pvParameters);#define TASK1_TASK_PRIO2#define TASK1_STK_SIZE 128  TaskHandle_t Task1Task_Handler;void task1_task(void *pvParameters);/*!    \brief      main function    \param[in]  none    \param[out] none    \retval     none*/int main(void){  nvic_priority_group_set(NVIC_PRIGROUP_PRE4_SUB0);    systick_config();My_GPIO_Init();    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();     while(1){    }}void start_task(void *pvParameters){    taskENTER_CRITICAL(); xTaskCreate((TaskFunction_t )task1_task, (const char*    )"task1_task",      (uint16_t)TASK1_STK_SIZE,   (void*   )NULL,      (UBaseType_t    )TASK1_TASK_PRIO,   (TaskHandle_t*  )&Task1Task_Handler);vTaskDelete(StartTask_Handler);     taskEXIT_CRITICAL();     }void task1_task(void *pvParameters){while(1){gpio_bit_write(GPIOA, GPIO_PIN_8, SET);    vTaskDelay(1000);gpio_bit_write(GPIOA, GPIO_PIN_8, RESET);vTaskDelay(1000);}}

测试视频

GD32F310移植FreeRTOS点灯实验

其他

感谢本次极术社区提供的开发板,这也是我第一次使用GD32,以前一直使用的都是STM32,其实国产的32板子非常不错。虽然STM32的生态确实是没话说,有CubeMX等工具,但也不是必需的。考虑到现在的STM32的价格,我觉得我们可以能够多多支持国产,MCU的生态也取决我们开发者,生态是靠我们一起建设。

推币机的世界