STM32学习笔记一一UCOSII(1)

STM32学习笔记一一UCOSII(1)1 简介 UCOSII 是一个可以基于 ROM 运行的 可裁减的 抢占式 实时多任务内核 具有高度可移植性 特别适合于微处理器和控制器 是和很多商业操作系统性能相当的实时操作系统 RTOS 1 1UCOSII 体系结构图 UCOSII 的移植 我们只需要修改 os cpu h os cpu a asm 和 os cpu c 等三个文件 os cpu h 进行数据类型的定义

前言:

为了方便查看博客,特意申请了一个公众号,附上二维码,有兴趣的朋友可以关注,和我一起讨论学习,一起享受技术,一起成长。

在这里插入图片描述


1. 简介

UCOSII 是一个可以基于 ROM 运行的、可裁减的、抢占式、实时多任务内核,具有高度可移植性,特别适合于微处理器和控制器,是和很多商业操作系统性能相当的实时操作系统(RTOS)。

1.1 UCOSII 体系结构图

这里写图片描述

UCOSII 的移植,我们只需要修改: os_cpu.h、 os_cpu_a.asm 和 os_cpu.c等三个文件。

os_cpu.h: 进行数据类型的定义,以及处理器相关代码和几个函数原型;

os_cpu_a.asm:是移植过程中需要汇编完成的一些函数,主要就是任务切换函数;

os_cpu.c:定义一些用户 HOOK 函数。

1.2 任务

任务:其实就是一个死循环函数,该函数实现一定的功能,一个工程可以有很多这样的任务(最多 255 个), UCOSII 对这些任务进行调度管理, 让这些任务可以并发工作(注意不是同时工作,并发只是各任务轮流占用 CPU,而不是同时占用,任何时候还是只有 1个任务能够占用 CPU), 这就是 UCOSII 最基本的功能。

Ucos 任务的一般格式为:

void MyTask (void *pdata) { 
     任务准备工作… While(1)//死循环 { 
     任务 MyTask 实体代码; OSTimeDlyHMSM(x,x,x,x);//调用任务延时函数,释放 cpu 控制权, } } 

1.3 任务优先级

ucos 中,每个任务都有唯一的一个优先级。优先级是任务的唯一标识。在 UCOSII中,使用 CPU 的时候,优先级高(数值小)的任务比优先级低的任务具有优先使用权,即任务就绪表中总是优先级最高的任务获得 CPU 使用权,只有高优先级的任务让出 CPU 使用权(比如延时)时,低优先级的任务才能获得 CPU 使用权。 UCOSII 不支持多个任务优先级相同,也就是每个任务的优先级必须不一样。

1.4 任务堆栈

存储器中的连续存储空间。为了满足任务切换和响应中断时保存 CPU 寄存器中的内容以及任务调用其他函数时的需要,每个任务都有自己的堆栈。在创建任务的时候,任务堆栈是任务创建的一个重要入口参数。

1.5 任务控制块 OS_TCB

用来记录任务堆栈指针,任务当前状态以及任务优先级等任务属性。UCOSII 的任何任务都是通过任务控制块(TCB)的东西来控制的,一旦任务创建了,任务控制块 OS_TCB 就会被赋值。每个任务管理块有 3 个最重要的参数:

1.任务函数指针;

2.任务堆栈指针;

3.任务优先级;

任务控制块就是任务在系统里面的身份证。

1.6 任务就绪表

用来记录系统中所有处于就绪状态的任务。它是一个位图,系统中每个任务都在这个图中占据一个进制位,该位置的状态(1 或者 0)就表示任务是否处于就绪状态。

1.7 任务调度

一是在任务就绪表中查找优先级最高的就绪任务,二是实现任务的切换。比如说,当一个任务释放 cpu 控制权后,进行一次任务调度,这个时候任务调度器首先要去任务就绪表查询优先级最高的就绪任务,查到之后,进行一次任务切换,转而去执行下一个任务。

1.8 状态切换

UCOSII 的每个任务都是一个死循环。每个任务都处在以下 5 种状态之一的状态下,这 5种状态是:睡眠状态、 就绪状态、 运行状态、 等待状态(等待某一事件发生)和中断服务状态。

睡眠状态: 任务在没有被配备任务控制块或被剥夺了任务控制块时的状态。

就绪状态: 系统为任务配备了任务控制块且在任务就绪表中进行了就绪登记,任务已经准备好了,但由于该任务的优先级比正在运行的任务的优先级低, 还暂时不能运行,这时任务的状态叫做就绪状态。

运行状态: 该任务获得 CPU 使用权,并正在运行中,此时的任务状态叫做运行状态。

等待状态: 正在运行的任务,需要等待一段时间或需要等待一个事件发生再运行时,该任务就会把 CPU 的使用权让给别的任务而使任务进入等待状态。

中断服务状态: 一个正在运行的任务一旦响应中断申请就会中止运行而去执行中断服务程序,这时任务的状态叫做中断服务状态。

5种状态之间的转换如下图:

这里写图片描述

2.UCOS相关函数

2.1 建立任务函数

如果想让 UCOSII 管理用户的任务,必须先建立任务。 UCOSII 提供了 2 个建立任务的函数: OSTaskCreat 和 OSTaskCreatExt,一般用 OSTaskCreat 函数来创建任务。

该函数原型为:

OSTaskCreate(void(*task)(void*pd),void*pdata,OS_STK*ptos,INTU prio)

task:是指向任务代码的指针;

pdata:是任务开始执行时,传递给任务的参数的指针;

ptos:是分配给任务的堆栈的栈顶指针;

prio :是分配给任务的优先级。

2.2 任务删除函数

任务删除,其实就是把任务置于睡眠状态。 UCOSII提供的任务删除函数原型为:

INT8U OSTaskDel(INT8U prio)

参数 prio :要删除的任务的优先级,可见该函数是通过任务优先级来实现任务删除的。

特别注意: 任务不能随便删除,必须在确保被删除任务的资源被释放的前提下才能删除!

2.3 请求任务删除函数

前面提到,必须确保被删除任务的资源被释放的前提下才能将其删除,所以通过向被删除任务发送删除请求,来实现任务释放自身占用资源后再删除。

UCOSII 提供的请求删除任务函数原型为:

INT8U OSTaskDelReq(INT8U prio)

通过优先级来确定被请求删除任务。

2.4 任务挂起函数

UCOSII 提供的任务挂起函数原型为:

INT8U OSTaskSuspend(INT8U prio)

2.5 任务恢复函数

UCOSII 提供的任务恢复函数原型为:

INT8U OSTaskResume(INT8U prio)

3. 移植 UCOSII

3.1 移植 UCOUS

3.2 编写任务函数并设置其堆栈大小和优先级等参数

编写任务函数,以便 UCOSII 调用。

设置函数堆栈大小,这个需要根据函数的需求来设置,如果任务函数的局部变量多,嵌套层数多,那么相应的堆栈就得大一些,如果堆栈设置小了,很可能出现的结果就是 CPU进入 HardFault,遇到这种情况,就必须把堆栈设置大一点了。另外,有些地方还需要注意堆栈字节对齐的问题,如果任务运行出现莫名其妙的错误(比如用到 sprintf 出错),请考虑是不是字节对齐的问题。

设置任务优先级, 这个需要根据任务的重要性和实时性设置,高优先级的任务有优先使用 CPU 的权利。

3.3 初始化 UCOSII,并在 UCOSII 中创建任务

调用 OSInit,初始化 UCOSII,通过调用 OSTaskCreate 函数创建我们的任务。

3.4 启动 UCOSII

调用 OSStart,启动 UCOSII。

4. 软件配置

4.1 UCOSII 源码说明

这里写图片描述

UCOSII-CORE:是UCOSII 的核心源码,不需要做任何变动。

UCOSII-PORT :移植 UCOSII 要修改的 3 个代码,这个在移植的时候完成。

UCOSII-CONFIG : UCOSII 的配置部分,主要由用户根据自己的需要对 UCOSII进行裁剪或其他设置。

4.2 UCOSII 简易测试

#include "led.h" #include "delay.h" #include "sys.h" #include "includes.h"  /UCOSII任务设置/// //START 任务 //设置任务优先级 #define START_TASK_PRIO 10 //开始任务的优先级设置为最低 //设置任务堆栈大小 #define START_STK_SIZE 64 //任务堆栈  OS_STK START_TASK_STK[START_STK_SIZE]; //任务函数 void start_task(void *pdata); //LED0任务 //设置任务优先级 #define LED0_TASK_PRIO 7  //设置任务堆栈大小 #define LED0_STK_SIZE 64 //任务堆栈  OS_STK LED0_TASK_STK[LED0_STK_SIZE]; //任务函数 void led0_task(void *pdata); //LED1任务 //设置任务优先级 #define LED1_TASK_PRIO 6  //设置任务堆栈大小 #define LED1_STK_SIZE 64 //任务堆栈 OS_STK LED1_TASK_STK[LED1_STK_SIZE]; //任务函数 void led1_task(void *pdata); int main(void) { 
     delay_init(); //延时函数初始化  NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);// 设置中断优先级分组2 LED_Init(); //初始化与LED连接的硬件接口 OSInit(); OSTaskCreate(start_task,(void *)0,(OS_STK *)&START_TASK_STK[START_STK_SIZE-1],START_TASK_PRIO );//创建起始任务 OSStart(); } //开始任务 void start_task(void *pdata) { 
     OS_CPU_SR cpu_sr=0; pdata = pdata; OS_ENTER_CRITICAL(); //进入临界区(无法被中断打断)  OSTaskCreate(led0_task,(void *)0,(OS_STK*)&LED0_TASK_STK[LED0_STK_SIZE-1],LED0_TASK_PRIO); OSTaskCreate(led1_task,(void *)0,(OS_STK*)&LED1_TASK_STK[LED1_STK_SIZE-1],LED1_TASK_PRIO); OSTaskSuspend(START_TASK_PRIO); //挂起起始任务. OS_EXIT_CRITICAL(); //退出临界区(可以被中断打断) } //LED0任务 void led0_task(void *pdata) { 
     while(1) { 
     LED0=0; delay_ms(80); LED0=1; delay_ms(920); }; } //LED1任务 void led1_task(void *pdata) { 
     while(1) { 
     LED1=0; delay_ms(300); LED1=1; delay_ms(300); }; } 

参考:

  1. 原子库函数手册
  2. 第3章 内核结构
版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/212324.html原文链接:https://javaforall.net

(0)
上一篇 2026年3月18日 下午8:18
下一篇 2026年3月18日 下午8:19


相关推荐

  • nginx菜鸟教程七

    nginx菜鸟教程七原理 浏览器 请求 gt 声明可以接受 gzip 压缩或 deflate 压缩或 compress 或 sdch 压缩从 http 协议的角度看请求头声明 acceopt encoding gzipdeflates 是指压缩算法 其中 sdch 是 google

    2026年3月17日
    2
  • iOS逆向之深入解析如何Hook所有+load方法及Category的处理

    iOS逆向之深入解析如何Hook所有+load方法及Category的处理一 类方法 loadiOS 四种方法可方便的在 premain 阶段执行代码 ObjectiveC 类的 load 方法 C staticinitia C C attribute constructor functions 动态库中的上面三种方法 所有类的 load 方法是在 main 函数之前 在主线程 以串行方式调用 因此任何一个 load 方法的耗时大小将直接影响到 App 的启动耗时 ObjectiveCRu

    2026年3月20日
    2
  • 2022.01.13 激活(JetBrains全家桶)

    (2022.01.13 激活)2021最新分享一个能用的的激活码出来,希望能帮到需要激活的朋友。目前这个是能用的,但是用的人多了之后也会失效,会不定时更新的,大家持续关注此网站~IntelliJ2021最新激活注册码,破解教程可免费永久激活,亲测有效,下面是详细链接哦~https://javaforall.net/100143.html…

    2022年3月31日
    178
  • 波特尔暗空分类法_老暗锁打不开了怎么办

    波特尔暗空分类法_老暗锁打不开了怎么办传说中的暗之连锁被人们称为 Dark。Dark 是人类内心的黑暗的产物,古今中外的勇者们都试图打倒它。经过研究,你发现 Dark 呈现无向图的结构,图中有 N 个节点和两类边,一类边被称为主要边,而另一类被称为附加边。Dark 有 N–1 条主要边,并且 Dark 的任意两个节点之间都存在一条只由主要边构成的路径。另外,Dark 还有 M 条附加边。你的任务是把 Dark 斩为不连通的两部分。一开始 Dark 的附加边都处于无敌状态,你只能选择一条主要边切断。一旦你切断了一条主要边,Dark

    2022年8月9日
    10
  • 这也许是史上最有趣的破解软件合集

    这也许是史上最有趣的破解软件合集一,破解版百度网盘,下载速度超快,公众号回复【h01】即可获取。二,微信点赞工具,使用例子:新开的饭店呀,要你出示什么100个赞就可以吃霸王餐呀,根本不用徒劳朋友圈,直接用这软件就行,给你装备的机会,好好表演,公众号回复【h02】即可获取。三,微信群二维码采集助手,这个给那些微商需要引流的就很有作用啦,普通人可能用不到,公众号回复【h03】即可获取四,一款便捷图章制作工具,不想用PS,那这款软件就…

    2022年5月27日
    55
  • tar压缩和解压文件或文件夹

    1. 使用tar压缩文件tar-zcvftest.tar.gz./test/该命令表示压缩当前文件夹下的文件夹test,压缩后缀名为test.tar.gz如果不需要压缩成gz,只需要后缀为tar格式的,那么输入如下命令:tar-cvftest.tar./test/ 2. 使用tar解压文件tar-xzvftest.tar.gz  该命令表示把后缀为….

    2022年4月3日
    90

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号