【GD32F427开发板试用】IAR 环境移植freertos

news2025/1/12 16:15:08

本篇文章来自极术社区与兆易创新组织的GD32F427开发板评测活动,更多开发板试用活动请关注极术社区网站。作者:andeyqi

freertos移植适配

社区之前已经有同学移植适配freertos,在GD32F427上跑了起来,之前的帖子是在MDK环境下适配的,本地的开发环境为IAR,准备在IAR环境下在板子上跑freertos,freertos 的内核文件相对很少,而且官方的代码下已经支持了CORTEX-M4架构,我们基本不用修改什么就能把官方的代码适配到GD32F427的板子上。
我们先简答看下freertos的代码目录结构(芯片架构相关的我们只关注IAR cortex-m4):

FreeRTOS\Source\
|-- croutine.c
|-- event_groups.c
|-- list.c
|-- queue.c
|-- readme.txt
|-- stream_buffer.c
|-- tasks.c
`-- timers.c
|-- include
|   |-- FreeRTOS.h
|   |-- StackMacros.h
|   |-- atomic.h
|   |-- croutine.h
|   |-- deprecated_definitions.h
|   |-- event_groups.h
|   |-- list.h
|   |-- message_buffer.h
|   |-- mpu_prototypes.h
|   |-- mpu_wrappers.h
|   |-- portable.h
|   |-- projdefs.h
|   |-- queue.h
|   |-- semphr.h
|   |-- stack_macros.h
|   |-- stdint.readme
|   |-- stream_buffer.h
|   |-- task.h
|   `-- timers.h
|-- portable
|   |-- IAR
|   |   |-- ARM_CM4F
|   |   |   |-- port.c
|   |   |   |-- portasm.s
|   |   |   `-- portmacro.h
|   |-- MemMang
|   |   |-- ReadMe.url
|   |   |-- heap_1.c
|   |   |-- heap_2.c
|   |   |-- heap_3.c
|   |   |-- heap_4.c
|   |   `-- heap_5.c

从上面的代码目录树看代码代码量还是不到的,一共需要的.c .s 文件一共10个文件左右,因为操作系统依赖的portable相关的代码官方的代码结构里已经有了,理论上把这些文件组织到工程内部编译通过操作系统就移植完成了。
MemMang 目录下freertos 创建任务及资源需要动态malloc 内存,需要支持内存管理的接口,根据实际的情况选择一个就行,本次移植使用的是heap_4.c文件。

移植过程

  1. 从freertos 下载官方最新源码(FreeRTOSv202112.00),加入上述源码文件及include路径加入工程。

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-veuU2spe-1670224270009)(https://pic3.zhimg.com/80/v2-d1cd669084a0320fe55d1e9b44b34c3e_1440w.webp)]

2.加入上述文件后编译会报如下error:
Error\[2\]: Failed to open #include file 'FreeRTOSConfig.h' C:\\Users\\Administrator\\Desktop\\GD32F427-VSTART\\GD32F4xx\_Demo\_Suites\_V2.6.1\\GD32427V\_START\_Demo\_Suites\\Projects\\02\_GPIO\_Key\_Polling\_mode\\FreeRTOSv202112.00\\FreeRTOS\\Source\\portable\\IAR\\ARM_CM4F\\portasm.s 29
freertos 内核是根据这个config文件进行配置的,我们从freertos的demo路径下选取个M4的开发板配置文件放进去,本次选取的是ST的F407的config文件(\FreeRTOSv202112.00\FreeRTOS\Demo\CORTEX_M4F_STM32F407ZG-SK)

3.添加完上述文件后继续 build 会发现会报如下的linkerror,因为freertos 和GD32 的固件库定义了这几个中段处理函数发生重复定义了,我们删除GD32固件库中的定义及freertos 依赖的回调hook函数,这些回调hook函数暂时我们用不到,直接从FreeRTOSConfig.h 中注释掉对应的hook.
删除gd32f4xx_it.c 中的
void PendSV_Handler(void)
void SysTick_Handler(void)
void SysTick_Handler(void)

并将从FreeRTOSConfig.h 中如下宏定义修改为0

#define configUSE_IDLE_HOOK                0
#define configUSE_TICK_HOOK                0
#define configCHECK_FOR_STACK_OVERFLOW    0
#define configUSE_MALLOC_FAILED_HOOK    0

4.至此已经可以编译通过,我们就可以创建任务来验证,freertos 能否正常运行了,因为freertos 源码中会根据配置的systick 时钟周期来配置systick 我们在main 函数里可以删除systick_config()的配置。
` /configure systick/

systick_config();`

5.修改测试代码创建两个任务1s周期打印,并将之前裸机环境的shell 做人一个任务在freertos中运行。

#include "gd32f4xx.h"
#include "systick.h"
#include <stdio.h>
#include <string.h>
#include "littleshell.h"
#include "FreeRTOS.h"
#include "task.h"


#define START_TASK_PRIO        1
#define START_STK_SIZE         128  
TaskHandle_t StartTask_Handler;
void start_task(void *pvParameters);

#define TASK1_TASK_PRIO        2
#define TASK1_STK_SIZE         128  
TaskHandle_t Task1Task_Handler;
void start_task1(void *pvParameters);


#define SHELL_TASK_PRIO        2
#define SHELL_STK_SIZE         256  
TaskHandle_t SHELLTask_Handler;

void start_task(void *pvParameters)
{
    while(1)
    {
        printf("I am task2\r\n" );
        vTaskDelay(1000);
    }

}


void start_task1(void *pvParameters)
{
    while(1)
    {
        printf("I am task1\r\n" );
        vTaskDelay(1000);
    }
}

int main(void)
{
    /* configure systick */
    systick_config();

    /* enable the LEDs GPIO clock */
    rcu_periph_clock_enable(RCU_GPIOC);

    /* configure LED1 GPIO port */
    gpio_mode_set(GPIOC, GPIO_MODE_OUTPUT, GPIO_PUPD_NONE, GPIO_PIN_6);
    gpio_output_options_set(GPIOC, GPIO_OTYPE_PP, GPIO_OSPEED_50MHZ, GPIO_PIN_6);
    /* reset LED1 GPIO pin */
    gpio_bit_reset(GPIOC, GPIO_PIN_6);
    
    /* enable GPIO clock */
    rcu_periph_clock_enable(RCU_GPIOB);

    /* enable USART clock */
    rcu_periph_clock_enable(RCU_USART0);

    /* configure USART0 TX as alternate function push-pull */
    gpio_mode_set(GPIOB, GPIO_MODE_AF, GPIO_PUPD_PULLUP, GPIO_PIN_6);
    gpio_output_options_set(GPIOB, GPIO_OTYPE_PP, GPIO_OSPEED_25MHZ, GPIO_PIN_6);

    /* configure USART0 RX as alternate function push-pull */
    gpio_mode_set(GPIOB, GPIO_MODE_AF, GPIO_PUPD_PULLUP, GPIO_PIN_7);
    gpio_output_options_set(GPIOB, GPIO_OTYPE_PP, GPIO_OSPEED_25MHZ, GPIO_PIN_7);
    
    /* configure the USART0 TX pin and USART0 RX pin */
    gpio_af_set(GPIOB, GPIO_AF_7, GPIO_PIN_6);
    gpio_af_set(GPIOB, GPIO_AF_7, GPIO_PIN_7);



    /* USART configure */
    usart_deinit(USART0);
    usart_baudrate_set(USART0, 115200U);
    usart_parity_config(USART0,USART_PM_NONE);
    usart_word_length_set(USART0,USART_WL_8BIT);
    usart_stop_bit_set(USART0,USART_STB_1BIT);
    usart_receive_config(USART0, USART_RECEIVE_ENABLE);
    usart_transmit_config(USART0, USART_TRANSMIT_ENABLE);
    
    usart_enable(USART0);

    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); 
    
    xTaskCreate((TaskFunction_t )start_task1,           
                (const char*    )"start_task",        
                (uint16_t       )TASK1_STK_SIZE,      
                (void*          )NULL,                  
                (UBaseType_t    )TASK1_TASK_PRIO,      
                (TaskHandle_t*  )&Task1Task_Handler); 
    
    xTaskCreate((TaskFunction_t )littleshell_main_entry,           
                (const char*    )"SHELLt_task",        
                (uint16_t       )SHELL_STK_SIZE,      
                (void*          )NULL,                  
                (UBaseType_t    )SHELL_TASK_PRIO,      
                (TaskHandle_t*  )&SHELLTask_Handler); 

    vTaskStartScheduler(); 
    //(void)littleshell_main_entry(NULL);
}


/* retarget the C library printf function to the USART */
int fputc(int ch, FILE *f)
{
    usart_data_transmit(USART0, (uint8_t)ch);
    while(RESET == usart_flag_get(USART0, USART_FLAG_TBE));
    return ch;
}

uint8_t uartgetchar(uint8_t * pdata)
{
    if(SET == usart_flag_get(USART0, USART_FLAG_RBNE))
    {
        *pdata = usart_data_receive(USART0);
    return 1;
    }
    else
        return 0;
}

unsigned int led(char argc,char ** argv)
{
    if(argc != 2)
        return 0;

    if(strcmp("on",argv[1]) == 0)
    {
        /* turn on LED1 */
        gpio_bit_set(GPIOC, GPIO_PIN_6);
    }
    else if(strcmp("off",argv[1]) == 0)
    {
        gpio_bit_reset(GPIOC, GPIO_PIN_6);
    }
    
    return 1;
}
LTSH_FUNCTION_EXPORT(led,"test led on/off");

shell 的实现可以参考如下上一篇帖子:
https://aijishu.com/a/1060000000367387

试验验证

将上述代码下载到板子运行,串口上周期的打印如下信息而且shell 也可以正常响应,说明任务已经正常的调度起来了,而且从log打印的时间戳查看也是每隔1s打印说明调度的周期配置的也是正确的。
代码地址如下:
https://github.com/andeyqi/GD32F427-VSTART

[2022-11-19 15:47:34.699] GD32#I am task1
[2022-11-19 15:47:34.715] I am task2
[2022-11-19 15:47:34.871] 
[2022-11-19 15:47:34.871] GD32#
[2022-11-19 15:47:35.043] GD32#
[2022-11-19 15:47:35.230] GD32#
[2022-11-19 15:47:35.402] GD32#
[2022-11-19 15:47:35.574] GD32#I am task1
[2022-11-19 15:47:35.699] I am task2
[2022-11-19 15:47:35.761] 
[2022-11-19 15:47:35.761] GD32#
[2022-11-19 15:47:35.917] GD32#
[2022-11-19 15:47:36.089] GD32#
[2022-11-19 15:47:36.261] GD32#
[2022-11-19 15:47:36.417] GD32#
[2022-11-19 15:47:36.574] GD32#I am task1
[2022-11-19 15:47:36.714] I am task2
[2022-11-19 15:47:36.745] 
[2022-11-19 15:47:36.745] GD32#
[2022-11-19 15:47:36.902] GD32#
[2022-11-19 15:47:37.073] GD32#
[2022-11-19 15:47:37.245] GD32#
[2022-11-19 15:47:37.433] GD32#
[2022-11-19 15:47:37.605] GD32#I am task1
[2022-11-19 15:47:37.698] I am task2
[2022-11-19 15:47:37.792] 
[2022-11-19 15:47:37.792] GD32#

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/62935.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

FITC-PEG-FA,Folic acid-PEG-Fluorescein,叶酸PEG荧光素

●外观以及性质&#xff1a; FITC-PEG-FA是一种绿色荧光叶酸-PEG衍生物&#xff0c;其激发/发射波长为~490 nm/~520 nm。荧光素标记的PEG叶酸是一种线性分子&#xff0c;叶酸连接到PEG链的一端&#xff0c;荧光素染料连接到另一个PEG端。叶酸也被称为维生素M、维生素B9或叶酸。…

什么?Coolbpf 不仅可以远程编译,还可以发现网络抖动!

近日&#xff0c;在 2022 云栖大会龙蜥峰会 eBPF & Linux 稳定性专场上&#xff0c;来自 eBPF 技术探索 SIG Maintainer 的毛文安分享了《Coolbpf 的应用实践》技术演讲&#xff0c;以下为本次演讲内容&#xff1a; 一、为什么要支持可移植&#xff1f; 随着 BPF 技术的发…

EL表达式 过滤器 监听器

这里写目录标题1 EL 表达式1.1 EL 表达式介绍1.2 EL 表达式快速入门1.3 EL 表达式获取数据1.4 EL 表达式注意事项1.5 EL表达式运算符1.6 EL 表达式使用细节1.7 EL 表达式隐式对象2 JSTL2.1 JSTL 介绍2.2 JSTL 核心标签库2.3 JSTL基本使用3 Filter3.1 过滤器介绍3.2 Filter 介绍…

很全很详细的GUI编程

为什么要学习&#xff1f; 写出自己想要的小工具。工作可能需要维护到swing界面。了解MVC架构&#xff0c;了解监听。 组件&#xff1a;窗口、弹窗、面板、文本框、列表框、按钮、图片、监听事件、鼠标、键盘事件、破解工具。 一、简介 GUI的核心技术&#xff1a;Swing AWT 因…

[附源码]计算机毕业设计基于springboot和vue的茶文化交流平台的设计与实现

项目运行 环境配置&#xff1a; Jdk1.8 Tomcat7.0 Mysql HBuilderX&#xff08;Webstorm也行&#xff09; Eclispe&#xff08;IntelliJ IDEA,Eclispe,MyEclispe,Sts都支持&#xff09;。 项目技术&#xff1a; SSM mybatis Maven Vue 等等组成&#xff0c;B/S模式 M…

路由规则——MVC控制器

MVC的路由规则配置方式比较多,咱们用得最多的是两种: A、全局规则。就是我们熟悉的”{controller}/{action}“。 app.MapControllerRoute(name: "bug",pattern: "{controller}/{action}"); app.MapControllerRoute(name: "八阿哥",pattern: &…

定积分的概念及可积条件

微积分是高等数学的核心&#xff0c;包含微分和积分。前面几篇我们介绍了微分及其逆运算——不定积分&#xff08;严格来说&#xff0c;不定积分属于微分模块&#xff09;。传送门&#xff1a;微分与导数 不定积分今天开始&#xff0c;我们进入积分模块。还是老样子&#xff0…

怎么关闭Windows安全启动?

什么是安全启动&#xff1f; 安全启动&#xff08;英文名为Secure Boot&#xff09;&#xff0c;是装有UEFI固件的较新PC上的一项重要安全功能&#xff0c;它可以保护计算机的启动过程&#xff0c;防止在启动过程中加载未签名的UEFI驱动程序和引导加载程序&#xff0c;以及阻止…

软件测试经典面试题:如何进行支付功能的测试?

非现金支付时代&#xff0c;非现金支付已经成为了生活不可或缺的一部分&#xff0c;我们只需要一台手机便可走遍全国各地&#xff08;前提是支付宝&#xff0c;微信有钱<00>&#xff09;,那么作为测试人员&#xff0c;支付测试也是非常重要的一环&#xff0c;那么下面我就…

Netty03——进阶

目录1.粘包与半包1.1.粘包现象1.2.半包现象1.3.现象分析1.4 解决方案1.4.1.短链接1.4.2.固定长度1.4.3.固定分隔符1.4.4.预设长度2.协议设计与解析2.1.为什么需要协议&#xff1f;2.2.redis 协议举例2.3.http 协议举例2.4.自定义协议要素2.5.编解码器3.聊天室案例3.1.聊天室业务…

Locust学习记录2-locust用户类学习【HttpUser,wait_time,weight和fixed_count属性】

HttpUser 每个模拟用户定义的类&#xff0c;都必须继承HttpUser&#xff0c;目的时为每个用户提供一个client属性&#xff0c;该属性是一个实例HttpSession&#xff0c;可用于向我们要进行负载测试的目标系统发出HTTP请求 当测试开始时&#xff0c;locust将为它模拟的每个用户…

Java内存模型

TOC Java内存模型 Java内存模型&#xff08;Java Memory Model&#xff0c;JMM&#xff09; 是《Java虚拟机规范》中定义的一种用来屏蔽各种硬件和操作系统的内存访问差异&#xff0c;以实现让Java程序在各种平台下都能达到一致性的内存访问效果的一种内存访问模型。从JDK&…

不要再稀里糊涂的使用反射了,四万字带你搞定Java反射(JDK8)

文章目录前言Java反射体系概述类型信息&#xff08;Type&#xff09;GenericDeclarationAnnotatedType成员信息&#xff08;Member&#xff09;属性信息&#xff08;Field&#xff09;构造函数信息&#xff08;Constructor&#xff09;方法信息&#xff08;Method&#xff09;修…

毕业设计-基于机器视觉的指针式仪表智能识别

目录 前言 课题背景和意义 实现技术思路 实现效果图样例 前言 &#x1f4c5;大四是整个大学期间最忙碌的时光,一边要忙着备考或实习为毕业后面临的就业升学做准备,一边要为毕业设计耗费大量精力。近几年各个学校要求的毕设项目越来越难,有不少课题是研究生级别难度的,对本科…

C++ 复制构造函数

在讲解 C 的复制构造函数之前这里先明确一个概念&#xff0c;C 的复制构造函数的意思并不是字面上的意思复制一个构造函数&#xff0c;而是有一种专门用于复制内容的构造函数被叫做复制构造函数。 复制构造函数对于 C 来说是非常重要的概念&#xff0c;所以我们必须掌握并牢记…

jsp设备信息查询系统Myeclipse开发mysql数据库web结构java编程计算机网页项目

一、源码特点 jsp设备信息查询系统 是一套完善的web设计系统&#xff0c;对理解JSP java编程开发语言有帮助&#xff0c;系统具有完整的源代码和数据库&#xff0c;系统主要采用B/S模式开发。开发环境为 TOMCAT7.0,Myeclipse8.5开发&#xff0c;数据库为Mysql&#xff0c;使用…

【算法笔记(六)】检索算法

算法笔记(六) 检索算法算法笔记(六)前言一、线性查找1.什么是线性查找2.需求规则3.人工图示演示4.代码实现二、二分查找1.什么是二分查找2.需求规则3.人工图示演示4.代码实现三.插值查找1.什么是插值查找2.需求规则3.人工图示演示4.代码实现四.斐波那契查找1.什么是斐波那契查找…

[附源码]计算机毕业设计基于springboot框架的资产管理系统设计与实现

项目运行 环境配置&#xff1a; Jdk1.8 Tomcat7.0 Mysql HBuilderX&#xff08;Webstorm也行&#xff09; Eclispe&#xff08;IntelliJ IDEA,Eclispe,MyEclispe,Sts都支持&#xff09;。 项目技术&#xff1a; SSM mybatis Maven Vue 等等组成&#xff0c;B/S模式 M…

TCP三次握手四次挥手及常见问题解决方案

一、重要概念 位码即tcp标志位,有6种表示: SYN(synchronous建立连接) ACK(acknowledgement 表示响应、确认) PSH(push表示有DATA数据传输) FIN(finish关闭连接) RST(reset表示连接重置) URG(urgent紧急指针字段值有效) 二、三次握手四次挥手流程图 三、三次握手 第一次握手&am…

一段有关MPU配置代码的几个小疑问

当我们阅读一些STM32F7或STM32H7系列芯片例程&#xff0c;或者基于这两类芯片通过cubeMx进行配置并用到MPU功能时&#xff0c;往往会在代码里看到下面这段MPU配置。对这段代码可能有人有些疑问&#xff0c;这里重点一起聊聊其中的3个&#xff0c;供参考。第一个疑问&#xff0c…