嵌入式RTOS编程

本文详细介绍了如何将嵌入式RTOS UCOSIII移植到STM32F103C8T6单片机上,包括UCOSIII的简介、CubeMX工程配置、源码准备、文件导入、串口发送、任务实现及实验结果。通过一系列步骤,读者可以学习到如何在STM32上搭建并运行多任务实时操作系统。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

一、UCOSIII介绍

UCOS是Micrium公司出品的RTOS类实时操作系统, UCOS目前有两个版本:UCOSII和UCOSIII。
UCOSIII是一个可裁剪、可剥夺型的多任务内核,而且没有任务数限制,提供了实时操作系统所需的所有功能,包括资源管理、同步、任务通信等。
UCOSIII是用C和汇编来写的,其中绝大部分都是用C语言编写的,只有极少数的与处理器密切相关的部分代码才是用汇编写的, UCOSIII结构简洁,可读性很强!非常适合初次接触嵌入式实时操作系统学生、嵌入式系统开发人员和爱好者学习。

二、CubeMX工程建立

2.1 新建工程

在这里插入图片描述

2.2 配置RCC

在这里插入图片描述

2.3 配置SYS

在这里插入图片描述

2.4 配置USART1串口

在这里插入图片描述

2.5 设置LED灯输出(PC13)

在这里插入图片描述

2.6 配置时钟源

在这里插入图片描述

2.7 创建工程

在这里插入图片描述
在这里插入图片描述

2.8 调试

生成代码后,在Keil中打开该工程,并在main函数中的while循环里添加语句:

while (1)
  {
    /* USER CODE END WHILE */

    /* USER CODE BEGIN 3 */
		HAL_GPIO_WritePin(GPIOC, GPIO_PIN_13, GPIO_PIN_SET);
		HAL_Delay(500);
		HAL_GPIO_WritePin(GPIOC, GPIO_PIN_13, GPIO_PIN_RESET);
		HAL_Delay(500);
  }
  /* USER CODE END 3 */

编译,将代码烧录到STM32F103中,BOOT0置零,按下Reset按钮,可以看到PC13LED灯闪烁,证明代码没有问题。

三、uCOSIII源码准备

进入 Micrium 公司官网下载中心:http://micrium.com/downloadcenter/
选择ST系列,点击 View all STMicroelectronics,点击 STMicroelectronics STM32F107,之后按照提示注册下载即可。

四、移植前的文件准备

4.1打开下载好的源码

在这里插入图片描述

4.2 将uCOS的5个相关文件复制到cubeMX工程的MDK-ARM文件夹下

在这里插入图片描述

五、开始移植

打开keil文件。

5.1 将uCOS文件添加到项目中

在这里插入图片描述

5.2 点击CPU–>Add Files…

MDK-ARM\uC-CPU路径下选中以下文件,Add添加。
在这里插入图片描述

5.3 点击LIB–>Add Files…

MDK-ARM\uC-LIB路径下选中下图红框文件,Add添加。
MDK-ARM\uC-LIB\Ports\ARM-Cortex-M3\RealView路径下选中下图绿框文件,Add添加。
在这里插入图片描述

5.4 点击PORT–>Add Files…

MDK-ARM\uCOS-III\Ports\ARM-Cortex-M3\Generic\RealView路径下选中以下文件,Add添加。
在这里插入图片描述

5.5 点击SOURCE–>Add Files…

MDK-ARM\uCOS-III\Source路径下选中以下全部 .c .h 文件,Add添加。
在这里插入图片描述

5.6 点击CONFIG–>Add Files…

MDK-ARM\uC-CONFIG路径下选中以下全部文件,Add添加。
在这里插入图片描述

5.7 点击BSP–>Add Files…

MDK-ARM\uC-BSP路径下选中以下全部文件,Add添加。
在这里插入图片描述
点击OK。

5.8 导入文件路径

按下图所示步骤操作即可
在这里插入图片描述

六、串口发送

6.1 为bsp.c和bsp.h文件添加代码

找到BSP下的bsp.c和bsp.h文件。
添加代码如下:
bsp.h

// bsp.h
#ifndef  __BSP_H__
#define  __BSP_H__

#include "stm32f1xx_hal.h"

void BSP_Init(void);

#endif

bsp.c

// bsp.c
#include "includes.h"

#define  DWT_CR      *(CPU_REG32 *)0xE0001000
#define  DWT_CYCCNT  *(CPU_REG32 *)0xE0001004
#define  DEM_CR      *(CPU_REG32 *)0xE000EDFC
#define  DBGMCU_CR   *(CPU_REG32 *)0xE0042004

#define  DEM_CR_TRCENA                   (1 << 24)
#define  DWT_CR_CYCCNTENA                (1 <<  0)

CPU_INT32U  BSP_CPU_ClkFreq (void)
{
    return HAL_RCC_GetHCLKFreq();
}

void BSP_Tick_Init(void)
{
	CPU_INT32U cpu_clk_freq;
	CPU_INT32U cnts;
	cpu_clk_freq = BSP_CPU_ClkFreq();
	
	#if(OS_VERSION>=3000u)
		cnts = cpu_clk_freq/(CPU_INT32U)OSCfg_TickRate_Hz;
	#else
		cnts = cpu_clk_freq/(CPU_INT32U)OS_TICKS_PER_SEC;
	#endif
	OS_CPU_SysTickInit(cnts);
}



void BSP_Init(void)
{
	BSP_Tick_Init();
	MX_GPIO_Init();
}


#if (CPU_CFG_TS_TMR_EN == DEF_ENABLED)
void  CPU_TS_TmrInit (void)
{
    CPU_INT32U  cpu_clk_freq_hz;


    DEM_CR         |= (CPU_INT32U)DEM_CR_TRCENA;                /* Enable Cortex-M3's DWT CYCCNT reg.                   */
    DWT_CYCCNT      = (CPU_INT32U)0u;
    DWT_CR         |= (CPU_INT32U)DWT_CR_CYCCNTENA;

    cpu_clk_freq_hz = BSP_CPU_ClkFreq();
    CPU_TS_TmrFreqSet(cpu_clk_freq_hz);
}
#endif


#if (CPU_CFG_TS_TMR_EN == DEF_ENABLED)
CPU_TS_TMR  CPU_TS_TmrRd (void)
{
    return ((CPU_TS_TMR)DWT_CYCCNT);
}
#endif


#if (CPU_CFG_TS_32_EN == DEF_ENABLED)
CPU_INT64U  CPU_TS32_to_uSec (CPU_TS32  ts_cnts)
{
	CPU_INT64U  ts_us;
  CPU_INT64U  fclk_freq;

 
  fclk_freq = BSP_CPU_ClkFreq();
  ts_us     = ts_cnts / (fclk_freq / DEF_TIME_NBR_uS_PER_SEC);

  return (ts_us);
}
#endif
 
 
#if (CPU_CFG_TS_64_EN == DEF_ENABLED)
CPU_INT64U  CPU_TS64_to_uSec (CPU_TS6
### C++在嵌入式实时操作系统(RTOS)中的开发与应用 #### 1. 嵌入式系统的特点及其对C++的支持 嵌入式系统的开发通常受到资源有限性的约束,因此需要一种既能提供高效性能又能简化复杂逻辑的语言。C++作为一种兼具高层抽象能力和低层硬件操作能力的编程语言,在嵌入式领域得到了广泛应用[^1]。 尽管如此,由于嵌入式环境可能缺乏完整的标准库支持以及运行时环境(Runtime),开发者需特别注意如何适配这些限制条件。例如,`operator new`作为C++的一个核心特性,并不依赖于特定的标准库实现,而是通过编译器内置机制完成动态内存分配功能[^3]。这意味着可以通过自定义方式将其链接到RTOS所提供的堆管理接口上,从而适应不同的目标平台需求。 #### 2. FreeRTOS与C++ FreeRTOS是一个轻量级的开源RTOS解决方案,主要用于微控制器级别的设备开发。虽然它的官方文档更倾向于推荐使用纯C来进行编码工作,但实际上它同样允许采用C++编写应用程序代码[^2]。然而值得注意的是,在混合模式下可能会遇到一些额外的技术难题,比如名称修饰(name mangling),这要求程序员采取适当措施解决此类兼容性问题。 对于希望利用C++优势(如封装、继承和多态等OOP概念)的同时又想保持良好移植性和效率的团队来说,则需要仔细权衡两者之间的利弊关系并制定相应的策略。 #### 3. 自定义new/delete运算符以集成RTOS堆管理 为了使C++程序能够在没有传统C++ runtime的情况下正常运作,特别是当涉及到动态存储分配的时候,可以考虑重载全局版本的新建(`::operator new`) 和删除 (`::operator delete`) 运算符。这样做的目的是让它们调用由具体RTOS提供的相应服务而不是默认的行为: ```cpp void* operator new(std::size_t size) { void *p = pvPortMalloc(size); // 使用FreeRTOS的pvPortMalloc代替malloc() if (p == NULL){ throw std::bad_alloc(); } return p; } void operator delete(void* ptr) noexcept{ vPortFree(ptr); // 使用FreeRTOS的vPortFree代替free() } ``` 上述方法展示了如何将C++的对象创建过程绑定至指定RTOS框架下的内存池管理系统之上。此技术不仅有助于减少不必要的开销还能够增强整个软件栈的安全保障水平。 #### 4. 结合实例探讨——基于Raspberry Pi 的嵌入式Linux+C++开发 除了传统的裸机或者RTOS驱动型设计之外,现代许多高端MCU/SoC都开始转向搭载精简版的操作系统如μCLinux 或者完整内核形式的GNU/Linux 。在这种情况下,《Embedded Linux Primer》这类教材就显得尤为重要了[^5]。书中提到的内容涵盖了从启动加载程序配置直到构建定制化rootfs镜像的所有环节;而对于那些想要进一步探索高级话题的人群而言,“固件开发项目1000例”专栏里的相关内容无疑会成为极好的参考资料之一[^4]。 综上所述,无论是在经典的RTOS场景还是新兴的嵌入式linux生态当中,合理运用c++都能够显著提升项目的可维护性和扩展潜力。 ---
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值