RT_Thread 临界区访问控制

本文介绍了RT_Thread操作系统中临界区访问控制的三种机制:全局中断禁止、禁止抢占以及使用互斥锁。分别阐述了每种机制的工作原理、优缺点,并通过代码示例进行说明,强调了在不同场景下选择合适锁机制的重要性。

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

一、WHY

   什么是临界区,为什么我们需要临界区的访问控制?首先来看下wiki对临界区的定义:


在同步的程序设计中,临界区段(Critical section)指的是一个访问共享资源(例如:共享设备或是共享存储器)的程序片段,而这些共享资源无法同时被多个线程访问的特性。当有线程进入临界区段时,其他线程或是进程必须等待,以确保这些共享资源是被异或的使用。比如打印机。


二、RT_Thread 临界区访问控制机制

2.1 全局中断:禁止全局中断,禁止抢占

全局中断开关也称为中断锁,是禁止多线程访问临界区最简单的一种方式,即通过关闭中断的方式,来保证当前线程不会被其他事件打断(因为整个系统已经不再响应那些可以触发线程重新调度的外部事件),也就是当前线程不会被抢占,除非这个线程主动放弃了处理器控制权,其函数接口如下:

rt_base_t rt_hw_interrupt_disable(void);
void rt_hw_interrupt_enable(rt_base_t level);

优点:使用中断锁来操作临界区的方法可以应用于任何场合,且其他几类同步方式都是依赖于中断锁而实现的,可以说中断锁是最强大的和最高效的同步方法。

缺点:在中断关闭期间系统将不再响应任何中断,也就不能响应外部的事件。所以中断锁对系统的实时性影响非常巨大,当使用不当的时候会导致系统完全无实时性可言。它属于粒度最大的访问控制机制

实现举例:

/*FILE: libcpu/arm/contex-m4/context_gcc.S*/
/*
 * rt_base_t rt_hw_interrupt_disable();
 */
.global rt_hw_interrupt_disable
.type rt_hw_interrupt_disable, %function
rt_hw_interrupt_disable:
    MRS     r0, PRIMASK
    CPSID   I
    BX      LR

/*
 * void rt_hw_interrupt_enable(rt_base_t level);
 */
.global rt_hw_interrupt_enable
.type rt_hw_interrupt_enable, %function
rt_hw_interrupt_enable:
    MSR     PRIMASK, r0
    BX      LR

2.2 使能全局中断,禁止抢占

在这种情况下,打开全局中断,关闭调度功能,因为禁止了线程的调度器,所以哪怕是ISR中使一些thread恢复为ready 状态,其他线程也仍然不能抢占当前进程的运行。因此,该机制可以用来禁止多thread间的临界区同步。

优点:相比2.1,不会禁止全局中断, ISR仍然可以得到处理。

缺点:因为禁止了抢占,因为ISR资源(锁,信号

### RT-Thread临界区的使用与实现 #### 一、什么是临界区 在多线程环境中,多个线程可能需要访问同一个共享资源或执行一段特定代码。为了防止数据竞争和不一致的状态发生,这段被保护的代码被称为 **临界区**[^2]。 #### 二、RT-Thread临界区保护机制 RT-Thread 提供了多种方法来保护临界区,主要包括通过关闭中断以及锁定系统调度器两种方式: 1. **关闭中断** - 使用 `rt_hw_interrupt_disable()` 函数关闭全局中断并获取当前中断状态。 - 在完成临界区操作后,调用 `rt_hw_interrupt_enable(level)` 恢复之前的中断状态[^5]。 此种方法适用于非常短时间内的临界区保护,因为长时间关闭中断会影响系统的实时性和响应能力。 2. **锁定系统调度器** - 调用 `rt_enter_critical()` 锁定系统调度器,阻止其他线程抢占 CPU 时间片。 - 当临界区操作完成后,调用 `rt_exit_critical()` 解锁调度器,恢复正常的任务切换逻辑[^3]。 这种方法不会影响硬件中断处理程序的正常运行,因此更适合于较复杂的场景。 #### 三、具体实现示例 ##### 方法一:基于中断控制的临界区保护 下面是一个简单的例子展示如何利用中断禁用来保护临界区: ```c #include <rtthread.h> void critical_section_example(void) { rt_base_t level; /* 关闭全局中断 */ level = rt_hw_interrupt_disable(); /* 执行临界区代码 */ // 访问共享资源的代码... /* 恢复全局中断 */ rt_hw_interrupt_enable(level); } int main(void) { // 初始化... while (1) { critical_section_example(); // 其他代码... } return 0; } ``` ##### 方法二:基于调度器锁定的临界区保护 另一种更推荐的方法是通过锁定调度器来进行临界区管理: ```c #include <rtthread.h> void thread_entry(void *parameter) { while (1) { /* 将线程调度器上锁 */ rt_enter_critical(); /* 以下进入临界区 */ // 对共享资源进行安全访问... /* 完成临界区操作后解锁调度器 */ rt_exit_critical(); // 继续执行其余部分... } } ``` 上述代码片段展示了如何在线程入口函数中正确地使用 `rt_enter_critical` 和 `rt_exit_critical` 来确保对共享资源的安全访问[^4]。 #### 四、注意事项 - 如果采用关闭中断的方式来保护临界区,则应尽量缩短临界区内代码的执行时间,以免降低整个系统的性能。 - 在大多数情况下,建议优先考虑使用调度器锁定而非直接操控中断开关,这样既能保障安全性又能维持较好的实时表现。 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值