临界段    
    代码的临界段也称为临界区,一旦这部分代码开始执行,则不允许任何中断打断。为确保临界段代码的执行不被中断,在进入临界段之前须关中断,而临界段代码执行完毕后,要立即开中断。 
    由于Cortex-M3/M4的RTX内核库中没有关闭中断的操作,也就是说RTX的源码中不存在临界段。

中断锁
    中断锁就是RTOS提供的开关中断函数,因为Cortex-M3/M4的RTX源码中没有关闭中断的操作,所以也就没有提供开关中断函数。 由于RTX没有提供开关中断函数,如果用户自己的应用代码需要开关中断的话怎么办呢?
    裸机时如何开关中断的,在使用了RTX后仍然使用以前的开关中断函数即可

任务锁
    为了防止当前任务的执行被其它高优先级的任务打断而提供的锁机制就是任务锁。实现任务锁可以通过给调度器加锁或者直接关闭RTOS内核定时器(就是前面一直说的系统滴答定时器)来实现。 
    1.通过给调度器加锁实现 给调度器加锁的话,就无法实现任务切换,高优先级任务也就无法抢占低优先级任务的执行,同时高优先级任务也是无法向低优先级任务切换的。像uCOS-II和uCOS-III是采用的这种方法实现任务锁。特别注意,这种方式只是禁止了调度器工作,并没有关闭任何中断。 
    2.通过关闭RTOS内核定时器实现 关闭了RTOS内核定时器的话,也就关闭了通过RTOS内核定时器中断实现任务切换的功能,因为在退出定时器中断时需要检测当前需要执行的最高优先级任务,如果有高优先级任务就绪的话需要做任务切换。RTX操作系统是采用的这种方式实现任务锁的。

os_resume

    1. #include <rtl.h>
    1. void os_resume (
    1. U32 sleep_time ); /* Number of ticks the system was in sleep mode. */

说明:

    该函数唤醒操作系统调度器. 客户在调用了os_suspend之后必须调用该方法来重新使能调度器.
    sleep_time参数标识系统将在休眠或者掉电模式下呆多久. 以系统周期作为测量依据.

返回值:

    无

注意要点:

    该函数只能在系统空闲任务下调用. 
    单系统处于power_down模式下,tick定时器将不再运行. 

例程:

    The wake-up timer, when expired, generates the interrupt and wakes-up the system. Hence, it must run also in power-down mode. The system resumes operation and needs to call the function os_resume(). This function restores the RTX and re-enables the OS task scheduler
    1. /* After Wake-up */
    1. sleep = (tc - LPC_WWDT->TV) / 250;
    1. }
    1. os_resume(sleep);


os_suspend

    1. #include <rtl.h>
    1. U32 os_suspend (void);

说明:

    该函数挂起操作系统调度器. 该函数将测量需要多久进入掉电模式并关闭操作系统调度器.当函数返回的时候,操作系统调度器就被挂起了
    对于RTX,当调用了os_suspend之后,就必须调用os_resume 来恢复系统调度.

返回值:

    无

注意要点:

    只能在空闲任务中调用该函数. 
    操作系统进入power_down模式.系统tick定时器被禁止. 

例程:

    1. #include <rtl.h>
    1. __task void os_idle_demon (void) {
    1. uint32_t sleep;
    1. SCB->SCR |= SCB_SCR_SLEEPDEEP_Msk; /* Configure Cortex-M3 for deep sleep */
    1. PWR->CR &= ~PWR_CR_PDDS; /* Enter Stop mode when in deepsleep */
    1. PWR->CR |= PWR_CR_LPDS; /* Voltage regulator in low-power */
    1. /* Enable LSI clock and wait until ready */
    1. RCC->CSR |= RCC_CSR_LSION;
    1. while ((RCC->CSR & RCC_CSR_LSIRDY) == 0);
    1. /* Enable power interface clock */
    1. RCC->APB1ENR |= RCC_APB1ENR_PWREN;
    1. /* Disable backup domain write protection */
    1. PWR->CR |= PWR_CR_DBP;
    1. /* Select LSI as clock source for RTC and enable RTC */
    1. RCC->BDCR &= ~RCC_BDCR_RTCSEL;
    1. RCC->BDCR |= RCC_BDCR_RTCSEL_1;
    1. RCC->BDCR |= RCC_BDCR_RTCEN;
    1. /* Disable the write protection for RTC registers */
    1. RTC->WPR = 0xCA;
    1. RTC->WPR = 0x53;
    1. /* Configure RTC auto-wakeup mode */
    1. RTC->ISR &= ~RTC_ISR_WUTF; /* Clear wakeup timer flag */
    1. RTC->CR &= ~RTC_CR_WUCKSEL; /* Set RTC clock to 2kHz */
    1. RTC->CR |= RTC_CR_WUTIE; /* Enable RTC wakeup timer interrupt */
    1. /* Configure EXTI line 22 for wakeup on rising edge */
    1. EXTI->EMR |= (1 << 22); /* Event request is not masked */
    1. EXTI->RTSR |= (1 << 22); /* Rising trigger enabled */
    1. NVIC_EnableIRQ (RTC_WKUP_IRQn); /* Enable RTC WakeUp IRQ */
    1. for (;;) {
    1. /* HERE: include optional user code to be executed when no task runs. */
    1. sleep = os_suspend (); /* OS Suspend */
    1. if (sleep) {
    1. RTC->ISR &= ~RTC_ISR_WUTF; /* Clear timer wakeup flag */
    1. RTC->CR &= ~RTC_CR_WUTE; /* Disable wakeup timer */
    1. while ((RTC->ISR & RTC_ISR_WUTWF) == 0);
    1. /* RTC clock is @2kHz, set wakeup time for OS_TICK >= 1ms */
    1. RTC->WUTR = (sleep * (OS_TICK / 1000) * 2);
    1. RTC->CR |= RTC_CR_WUTE; /* Enable wakeup timer */
    1. __WFE (); /* Enter STOP mode */
    1. /* After Wake-up */
    1. if ((RTC->ISR & RTC_ISR_WUTF) == 0) {
    1. sleep = 0; /* We didn't enter Stop mode */
    1. }
    1. }
    1. os_resume (sleep); /* OS Resume */
    1. }
    1. }

tsk_lock

    1. #include <rtl.h>
    1. void tsk_lock (void);

说明:

    该函数禁止RTX内核时间中断,也就自然禁止了操作系统调度器.

返回值:

    无

注意要点:

    不能在中断向量中调用该函数. 
    不能在中断处理程序中调用该函数. 
    当禁用了内核时间中断,操作系统时间中断和时间片轮转中断被禁止,超时功能不在工作. 因此,强烈建议关RTX内核定时器中断的时间越短越好. 

例程:

    1. #include <rtl.h>
    1. void protect_critical_op () {
    1. tsk_lock ();
    1. do_critical_op ();
    1. tsk_unlock ();
    1. }

tsk_unlock

    1. #include <rtl.h>
    1. void tsk_unlock (void);

说明:

   函数tsk_unlock用于使能RTX内核定时器中断,因此也就重新开启任务切换。注意tsk_unlock一定要跟tsk_lock配套使用.

返回值:

    无

注意要点:

    函数tsk_lock不支持嵌套调用. 
    不允许在中断服务程序中调用tsk_lock.

例程:

    1. #include <rtl.h>
    1. void protect_critical_op () {
    1. tsk_lock ();
    1. do_critical_op ();
    1. tsk_unlock ();
    1. }

















RTX临界段,中断锁与任务锁的更多相关文章

  1. Linux——临界段,信号量,互斥锁,自旋锁,原子操作

    一. linux为什么需要临界段,信号量,互斥锁,自旋锁,原子操作? 1.1. linux内核后期版本是支持多核CPU以及抢占式调度.这里就存在一个并发,竞争状态(简称竟态). 1.2. 竞态条件 发 ...

  2. RTX——第11章 临界段,任务锁和中断锁

    以下内容转载自安富莱电子: http://forum.armfly.com/forum.php 临界段代码的临界段也称为临界区,一旦这部分代码开始执行,则不允许任何中断打断.为确保临界段代码的执行不被 ...

  3. FreeRTOS不允许在中断服务程序和临界段中执行不确定的性的操作

    举例 等待事件标志组的任务,要是在中断服务程序中设置事件标志组,但不知道当前有多少个任务在等待此事件标志,这个操作即为不确定性操作,为了不在中断服务程序中执行此不确定性操作,只在中断服务程序中给一确定 ...

  4. FreeRTOS 调度锁,任务锁和中断锁

    以下转载自安富莱电子: http://forum.armfly.com/forum.php 调度锁调度锁就是 RTOS 提供的调度器开关函数,如果某个任务调用了调度锁开关函数,处于调度锁开和调度锁关之 ...

  5. JAVA锁机制-可重入锁,可中断锁,公平锁,读写锁,自旋锁,

    如果需要查看具体的synchronized和lock的实现原理,请参考:解决多线程安全问题-无非两个方法synchronized和lock 具体原理(百度) 在并发编程中,经常遇到多个线程访问同一个 ...

  6. FreeRTOS 临界段和开关中断

    以下转载自安富莱电子: http://forum.armfly.com/forum.php 临界段代码的临界段也称为临界区,一旦这部分代码开始执行,则不允许任何中断打断.为确保临界段代码的执行不被中断 ...

  7. FreeRTOS 中断配置和临界段

    中断屏蔽寄存器 PRIMASK.FAULTMASK和BASEPRI 1.PRIMASK:这是个只有1个位的寄存器.当它置1时, 就关掉所有可屏蔽的异常,只剩下 NMI和硬fault可以响应.它的缺省值 ...

  8. 8.0-uC/OS-III临界段

    1.临界段 (临界段代码,也叫临界区,是指那些必须完整连续运行,不可被打断的代码段) 锁调度器,可以执行ISR,开启调度器不可执行ISR: (1).临界段代码,也称作临界域,是一段不可分割的代码. u ...

  9. jvm高级特性(6)(线程的种类,调度,状态,安全程度,实现安全的方法,同步种类,锁优化,锁种类)

    JVM高级特性与实践(十三):线程实现 与 Java线程调度 JVM高级特性与实践(十四):线程安全 与 锁优化 一. 线程的实现 线程其实是比进程更轻量级的调度执行单位. 线程的引入,可以把一个检查 ...

随机推荐

  1. Android Studio 设置字体

    File->Settings->Editor->Colors & Fonts->Font->Editor Font

  2. 模板-gcd

    GCD int gcd(int a, int b) { return b == 0 ? a : gcd(b, a%b); } EXGCD void ex_gcd(int a, int b, int & ...

  3. vscode rn代码格式化配置

    安装Beautify和React-beautify扩展程序,并在项目根目录创建配置文件.jsbeautifyrc,并复制下面代码到此配置文件里: { "mode": "b ...

  4. 【翻译】Spark 调优 (Tuning Spark) 中文版

    由于Spark自己的调优guidance已经覆盖了很多很有价值的点,因此这里直接翻译一份过来.也作为一个积累. Spark 调优 (Tuning Spark) 由于大多数Spark计算任务是在内存中运 ...

  5. GMA Round 1 函数求值

    传送门 函数求值 设函数$f(x)=x^{2018}+a_{2017}*x^{2017}+a_{2016}*x^{2016}+...+a_{2}*x^2+a_{1}*x+a_{0}$,其中$a_{0} ...

  6. [原创]RedisDesktopManager工具使用介绍

    [原创]RedisDesktopManager工具使用介绍 1 RedisDesktopManager简介 一款能够跨平台使用的开源性redis可视化工具.redis desktop manager主 ...

  7. Deepin 15.4 升级 chrome flash

    到 adobe 官方下载 flash插件 flash_player_ppapi_linux ~/.config/google-chrome/PepperFlash下建个目录 23.0.0.185,把 ...

  8. Mac上搭建ELK

    转载自我的个人博客:http://blog.ywheel.cn/post/2017/03/04/setup_elk_on_mac/ 最近的项目需要对文本数据各字段进行快速检索.组合查询.模糊查询,在架 ...

  9. Centos PHP+Apache执行exec()等Linux脚本权限设置的详细步骤

    1. 查看一下你的Apache的执行用户是谁: lsof -i:80         运行之后的结果为: 从图中我们可以清楚的看到,httpd(也就是Apache)的执行用户为:exec_shell( ...

  10. [Python设计模式] 第27章 正则表达式——解释器模式

    github地址:https://github.com/cheesezh/python_design_patterns 解释器模式 解释器模式,给定一个语言,定一个它的文法的一种表示,并定一个一个解释 ...