UCOS-III中断、临界段代码保护、任务延时

中断
应内部或外部异步事件的请求中止当前任务,而去处理异步事件所要求的任务的过程叫做中断 UCOSIII中的中断服务函数,例如:
void USART1_IRQHandler(void)
{
    OSIntEnter();
    //xxxxxxxxxxx 一些操作
    OSIntExit();
}
进入和退出中断服务函数
进入中断服务函数使用函数 OSIntEnter()
OSIntNestingCtr 来记录中断嵌套次数, UCOSIII 最多支持 250 级的中断嵌套。
退出中断服务函数时调用 OSIntExit()
UCOSIII 临界段代码保护
临界段代码也叫临界区,是指那些必须完整连续运行,不可被打断的代码段,当访问这些临界段代码的时候需要对这些临界段代码进行保护。
当宏 OS_CFG_ISR_POST_DEFERRED_EN 0 时, UCOSIII 使用关中断的方式来保护临界段的代码, 当设置为1 时,会采用给调度器上锁的方式来保护临界段代码。
UCOSIII 定义了一个进入临界段代码宏: OS_CRITICAL_ENTER()
定义了两个退出临界段代码的宏: OS_CRITICAL_EXIT OS_CRITICAL_EXIT_NO_SCHED()
任务延时:
UCOSIII 中的任务是一个无线循环并且还是一个抢占式内核,为了使高优先级的任务不至于独占 CPU ,可以给其他优先级较低的任务获取CPU 使用权UCOSIII中除空闲任务外,所有任务必须在合适的位置调用系统提供的延时函数,让当前的任务暂停运行一段时间并进行一个任务切换。
延时函数有两种, OSTimeDly() OSTimeDlyHMSM()
OSTimeDly() 有三种工作模式,相对模式、周期模式和绝对模式
OSTimeDlyHMSM() 仅在相对模式下工作
取消任务的延时
延时任务可以通过在其他任务中调用函数 OSTimeDlyResume() 取消延时而进入就绪状态,此函数最后会引发一次任务调度。
获取和设置系统时间:
UCOSIII 定义了一个 CPU_INT32U 类型的全局变量 OSTickCtr 来记录系统时钟节拍数,在嗲用 OSInit() 时被初始化为0 ,以后每发生一个时钟节拍, OSTickCtr 1
OSTimeSet() 允许用户改变当前时钟节拍计数器的值,慎用!!!
OSTimeGet() 用来获取动迁时钟节拍计数器的值。
#include "led.h"
#include "delay.h"
#include "Sys.h"
#include "Usart.h"
#include "includes.h"
#include "os_app_hooks.h"

// 任务优先级
#define START_TASK_PRIO 3
//任务队列大小
#define START_STK_SIZE 128
// 任务控制块
OS_TCB StartTaskTCB;
// 任务堆栈
CPU_STK START_TASK_STK[START_STK_SIZE];
// 任务函数
void start_task(void *p_arg);
// 任务优先级
#define TASK1_TASK_PRIO 4
//任务队列大小
#define TASK1_STK_SIZE 128
// 任务控制块
OS_TCB Task1_TaskTCB;
// 任务堆栈
CPU_STK TASK1_TASK_STK[TASK1_STK_SIZE];
// 任务函数
void task1_task(void *p_arg);
// 任务优先级
#define TASK2_TASK_PRIO 4
//任务队列大小
#define TASK2_STK_SIZE 128
// 任务控制块
OS_TCB Task2_TaskTCB;
// 任务堆栈
CPU_STK TASK2_TASK_STK[TASK2_STK_SIZE];
// 任务函数
void task2_task(void *p_arg);

int main(void)
{
    OS_ERR err;
    CPU_SR_ALLOC();
    delay_init();
    NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);
    uart_init(115200);
    LED_Init();
    OSInit(&err); // ucosiii初始化
    OS_CRITICAL_ENTER(); // 进入临界区
    OSTaskCreate(
        (OS_TCB * )&StartTaskTCB,
        (CPU_CHAR * ) "start task",
        (OS_TASK_PTR ) start_task,
        (void * ) 0,
        (OS_PRIO ) START_TASK_PRIO,
        (CPU_STK * ) &START_TASK_STK[0],
        (CPU_STK_SIZE) START_STK_SIZE / 10,
        (CPU_STK_SIZE) START_STK_SIZE,
        (OS_MSG_QTY ) 0,
        (OS_TICK ) 0,
        (void * ) 0,
        (OS_OPT ) OS_OPT_TASK_STK_CHK | OS_OPT_TASK_STK_CLR,
        &err
    );
    OS_CRITICAL_EXIT(); // 退出临界区
    OSStart(&err);
    while(1);
}
void start_task(void *p_arg)
{
    OS_ERR err;
    CPU_SR_ALLOC(); // 如果要使用临界区保护,那么必须调用CPU_SR_ALLOC();
    p_arg = p_arg;
    CPU_Init();
#if OS_CFG_STAT_TASK_EN > 0u
    OSStatTaskCPUUsageInit(&err); //统计任务
#endif
#ifdef CPU_CFG_INT_DIS_MEAS_EN //如果使能了测量中断关闭时间
    CPU_IntDisMeasMaxCurReset();
#endif
#if OS_CFG_SCHED_ROUND_ROBIN_EN //当使用时间片轮转的时候
    //使能时间片轮转调度功能,时间片长度为1个系统时钟节拍,既1*5=5ms
    OSSchedRoundRobinCfg(DEF_ENABLED,1,&err);
#endif
#if OS_CFG_APP_HOOKS_EN
    App_OS_SetAllHooks();
#endif
    OS_CRITICAL_ENTER(); //进入临界区
    //创建TASK1任务
    OSTaskCreate((OS_TCB * )&Task1_TaskTCB,
        (CPU_CHAR * )"Task1 task",
        (OS_TASK_PTR )task1_task,
        (void * )0,
        (OS_PRIO )TASK1_TASK_PRIO,
        (CPU_STK * )&TASK1_TASK_STK[0],
        (CPU_STK_SIZE)TASK1_STK_SIZE/10,
        (CPU_STK_SIZE)TASK1_STK_SIZE,
        (OS_MSG_QTY )0,
        (OS_TICK )3, // 时间片长度3 x 5 = 15毫秒
        (void * )0,
        (OS_OPT )OS_OPT_TASK_STK_CHK|OS_OPT_TASK_STK_CLR,
        (OS_ERR * )&err
    );
    //创建TASK2任务
    OSTaskCreate((OS_TCB * )&Task2_TaskTCB,
        (CPU_CHAR * )"task2 task",
        (OS_TASK_PTR )task2_task,
        (void * )0,
        (OS_PRIO )TASK2_TASK_PRIO,
        (CPU_STK * )&TASK2_TASK_STK[0],
        (CPU_STK_SIZE)TASK2_STK_SIZE/10,
        (CPU_STK_SIZE)TASK2_STK_SIZE,
        (OS_MSG_QTY )0,
        (OS_TICK )3,
        (void * )0,
        (OS_OPT )OS_OPT_TASK_STK_CHK|OS_OPT_TASK_STK_CLR,
        (OS_ERR * )&err);
        OS_CRITICAL_EXIT(); //退出临界区
        OSTaskDel((OS_TCB*)0,
        &err
    ); //删除start_task任务自身
}
void task1_task(void * p_arg)
{
    u8 i,task1_num=0;
    OS_ERR err;
    p_arg = p_arg;
    while(1)
    {
        task1_num++; //任务1执行次数加1 注意task1_num1加到255的时候会清零!!
        for(i=0;i<5;i++) printf("Task1:01234\r\n");
        LED0 = ~LED0;
        OSTimeDlyHMSM(0,0,1,0,OS_OPT_TIME_HMSM_STRICT,&err); //延时1s
    }
}
void task2_task(void * p_arg)
{
    u8 i,task2_num=0;
    OS_ERR err;
    p_arg = p_arg;
    while(1)
    {
        task2_num++; //任务2执行次数加1 注意task1_num2加到255的时候会清零!!
        for(i=0;i<5;i++) printf("Task2:56789\r\n");
        LED1 = ~LED1;
        OSTimeDlyHMSM(0,0,1,0,OS_OPT_TIME_HMSM_STRICT,&err); //延时1s
    }
}
    

  • 5
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
uCOS-III是一款实时操作系统,可以支持多任务并行执行。在uCOS-III中,创建任务需要使用OSTaskCreate函数。下面是一个简单的任务创建实验代码分析: ```c #include <includes.h> #define TASK_STK_SIZE 512 static OS_TCB Task1TCB; static CPU_STK Task1Stk[TASK_STK_SIZE]; static void Task1(void *p_arg); static OS_TCB Task2TCB; static CPU_STK Task2Stk[TASK_STK_SIZE]; static void Task2(void *p_arg); int main(void) { OS_ERR err; OSInit(&err); OSTaskCreate((OS_TCB *)&Task1TCB, (CPU_CHAR *)"Task1", (OS_TASK_PTR )Task1, (void *)0, (OS_PRIO )1, (CPU_STK *)&Task1Stk[0], (CPU_STK_SIZE)TASK_STK_SIZE/10, (CPU_STK_SIZE)TASK_STK_SIZE, (OS_MSG_QTY )0, (OS_TICK )0, (void *)0, (OS_OPT )(OS_OPT_TASK_STK_CHK | OS_OPT_TASK_STK_CLR), (OS_ERR *)&err); OSTaskCreate((OS_TCB *)&Task2TCB, (CPU_CHAR *)"Task2", (OS_TASK_PTR )Task2, (void *)0, (OS_PRIO )2, (CPU_STK *)&Task2Stk[0], (CPU_STK_SIZE)TASK_STK_SIZE/10, (CPU_STK_SIZE)TASK_STK_SIZE, (OS_MSG_QTY )0, (OS_TICK )0, (void *)0, (OS_OPT )(OS_OPT_TASK_STK_CHK | OS_OPT_TASK_STK_CLR), (OS_ERR *)&err); OSStart(&err); return 0; } static void Task1(void *p_arg) { while(1) { printf("Task1 running...\n"); OSTimeDlyHMSM(0, 0, 1, 0, OS_OPT_TIME_HMSM_STRICT, NULL); } } static void Task2(void *p_arg) { while(1) { printf("Task2 running...\n"); OSTimeDlyHMSM(0, 0, 1, 0, OS_OPT_TIME_HMSM_STRICT, NULL); } } ``` 在这个实验代码中,首先定义了两个任务Task1和Task2,每个任务都是一个无限循环,每隔1秒输出一次任务名。 在main函数中,首先调用OSInit函数初始化uCOS-III,然后使用OSTaskCreate函数创建任务。对于每个任务,需要指定任务控制块(TCB)、任务名、任务入口函数、任务参数、任务优先级、任务栈、任务栈大小、任务消息队列大小、任务定时器滴答数、任务扩展参数、任务选项和错误指针。在这个例子中,我们创建了两个任务Task1和Task2,分别使用优先级1和2,任务栈大小为512,任务栈使用数组Task1Stk和Task2Stk,任务入口函数分别为Task1和Task2。 最后,调用OSStart函数启动uCOS-III系统。在系统启动后,任务会自动并发执行,输出任务名。 需要注意的是,创建任务时需要根据实际应用场景合理设置任务优先级和任务栈大小,以免出现任务堆栈溢出或优先级反转等问题。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

柠檬酸田

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值