当前位置: 首页 > wzjs >正文

天津哪家公司做企业网站企业展厅设计公司哪家好怎么样

天津哪家公司做企业网站,企业展厅设计公司哪家好怎么样,建设网站专家,北京餐饮品牌设计公司临界区 在 FreeRTOS 中,临界区(Critical Section) 是指程序中一段必须以原子方式执行的代码区域,在此区域内不允许发生任务切换或中断干扰,以保护共享资源或执行关键操作。FreeRTOS 提供了多种机制来实现临界区&#…

临界区

在 FreeRTOS 中,临界区(Critical Section) 是指程序中一段必须以原子方式执行的代码区域,在此区域内不允许发生任务切换或中断干扰,以保护共享资源或执行关键操作。FreeRTOS 提供了多种机制来实现临界区,下面详细介绍其原理、实现和应用场景。

一、临界区的核心机制

FreeRTOS 的临界区主要通过 中断屏蔽 和 调度器挂起 两种方式实现:

1. 基于中断屏蔽的临界区
  • 原理:通过操作 Cortex-M 处理器的 BASEPRI 或 PRIMASK 寄存器,临时提升当前执行优先级,屏蔽低优先级中断。
  • 特点
    • 轻量级:开销小,适用于短时间保护。
    • 范围可控:默认只屏蔽优先级 ≤ configLIBRARY_MAX_SYSCALL_INTERRUPT_PRIORITY 的中断(通常为 5)。
2. 基于调度器挂起的临界区
  • 原理:通过增加 uxSchedulerSuspended 计数器,暂停任务调度器,防止任务切换。
  • 特点
    • 不影响中断:仅阻止任务切换,中断仍可响应。
    • 适用于长时间操作:如文件系统操作、复杂计算。

二、FreeRTOS 临界区 API

FreeRTOS 提供了两组临界区 API,分别用于 中断安全 和 非中断安全 场景:

1. 非中断安全 API
// 进入临界区(基于 BASEPRI 或 PRIMASK)
taskENTER_CRITICAL();// 临界区代码...// 退出临界区
taskEXIT_CRITICAL();
  • 实现机制
    在 Cortex-M 内核中,默认通过设置 BASEPRI 寄存器屏蔽低优先级中断(如优先级 ≤ 5),允许高优先级中断(如定时器、通信中断)继续执行。
2. 中断安全 API(用于 ISR)
// 在中断服务程序中进入临界区
uint32_t ulOriginalInterruptStatus = taskENTER_CRITICAL_FROM_ISR();// 临界区代码...// 退出临界区,恢复中断状态
taskEXIT_CRITICAL_FROM_ISR(ulOriginalInterruptStatus);
  • 注意事项
    该 API 仅在中断优先级 ≤ configLIBRARY_MAX_SYSCALL_INTERRUPT_PRIORITY 时有效,否则需使用专用的中断安全机制(如信号量的 FromISR 版本)。

三、临界区的实现源码分析

以下是 FreeRTOS 中临界区的核心实现(以 Cortex-M 为例):

1. taskENTER_CRITICAL () 的实现
// tasks.c
#define taskENTER_CRITICAL()             portENTER_CRITICAL()// portmacro.h(Cortex-M 实现)
#define portENTER_CRITICAL()             vPortEnterCritical()void vPortEnterCritical( void )
{__asm volatile("    cpsid i                         \n"  // 禁用中断(PRIMASK=1)"    ldr r0, =uxCriticalNesting      \n"  // 加载临界区嵌套计数器"    ldr r1, [r0]                    \n""    add r1, r1, #1                  \n"  // 计数器加1"    str r1, [r0]                    \n""    cmp r1, #1                      \n"  // 检查是否首次进入"    bne skip_basepri_set            \n""    ldr r0, =configKERNEL_INTERRUPT_PRIORITY \n"  // 加载 BASEPRI 值(如 0x50)"    msr basepri, r0                 \n"  // 设置 BASEPRI"skip_basepri_set:                   \n""    cpsie i                         \n"  // 重新启用中断(PRIMASK=0): : : "r0", "r1", "memory");
}
  • 关键点
    • 首次进入时设置 BASEPRI(如 0x50),屏蔽优先级 ≤ 5 的中断。
    • 嵌套进入时仅增加计数器,不重复设置 BASEPRI,减少开销。
2. taskEXIT_CRITICAL () 的实现
void vPortExitCritical( void )
{__asm volatile("    cpsid i                         \n"  // 禁用中断"    ldr r0, =uxCriticalNesting      \n"  // 加载嵌套计数器"    ldr r1, [r0]                    \n""    subs r1, r1, #1                 \n"  // 计数器减1"    str r1, [r0]                    \n""    bne skip_basepri_clear          \n"  // 非最后一次退出,跳过"    mov r0, #0                      \n"  // 准备清零 BASEPRI"    msr basepri, r0                 \n"  // 清零 BASEPRI"skip_basepri_clear:                 \n""    cpsie i                         \n"  // 重新启用中断: : : "r0", "r1", "memory");
}
  • 关键点
    最后一次退出时才清零 BASEPRI,确保嵌套临界区的正确性。

四、临界区与关中断的区别

特性临界区(taskENTER_CRITICAL)关中断(taskDISABLE_INTERRUPTS)
屏蔽范围仅屏蔽优先级 ≤ configMAX_SYSCALL_INTERRUPT_PRIORITY 的中断屏蔽所有可屏蔽中断
高优先级中断允许执行(如定时器、通信中断)被屏蔽
嵌套支持自动支持(通过计数器)需手动管理状态
执行时间较长(涉及寄存器操作)极短(单周期指令)
FreeRTOS 推荐场景常规临界区保护极短时间的关键操作(如调度器切换)

五、使用注意事项

  1. 临界区应尽量短小
    长时间占用会影响系统响应性,尤其在高优先级中断被屏蔽时。

  2. 禁止在临界区内调用阻塞 API
    如 vTaskDelay()xQueueReceive() 等,可能导致调度器卡死。

  3. 中断服务程序(ISR)中的临界区

    • 若 ISR 优先级 ≤ configLIBRARY_MAX_SYSCALL_INTERRUPT_PRIORITY,可安全使用。
    • 若 ISR 优先级更高(如 4),使用 taskENTER_CRITICAL_FROM_ISR() 变体。
  4. 与调度器挂起的配合
    若需要长时间保护资源且不影响中断,可结合使用 vTaskSuspendAll() 和 xTaskResumeAll()

六、典型应用场景

1. 保护共享资源
static uint32_t sharedResource = 0;void updateResource( void )
{taskENTER_CRITICAL();  // 进入临界区sharedResource++;      // 修改共享资源taskEXIT_CRITICAL();   // 退出临界区
}
2. 执行原子操作
void atomicOperation( void )
{taskENTER_CRITICAL();// 执行不可分割的操作(如配置外设寄存器)PERIPHERAL->CONTROL = 0x1234;taskEXIT_CRITICAL();
}
3. 嵌套临界区示例
void nestedCriticalSection( void )
{taskENTER_CRITICAL();  // 首次进入:设置 BASEPRI// 临界区1taskENTER_CRITICAL();  // 嵌套进入:仅增加计数器// 临界区2taskEXIT_CRITICAL();  // 嵌套退出:仅减少计数器// 临界区1taskEXIT_CRITICAL();  // 最后退出:清零 BASEPRI
}

七、总结

FreeRTOS 的临界区机制通过 智能管理 BASEPRI 寄存器 和 嵌套计数器,实现了高效且安全的中断屏蔽。其核心优势在于:

  • 选择性屏蔽:仅屏蔽必要的低优先级中断,保留高优先级中断响应能力。
  • 自动嵌套:无需手动管理中断状态,避免常见的编程错误。
  • 轻量级开销:通过寄存器操作而非锁机制,适合实时系统。

合理使用临界区是保证嵌入式系统 数据一致性 和 实时性 的关键。

 任务调度器挂起与恢复

在 FreeRTOS 中,挂起任务调度器(Suspend Scheduler) 是一种暂停任务切换的机制,允许当前执行的任务在不被其他任务抢占的情况下连续运行。以下是其核心原理、实现和应用场景的详细解析:

一、任务调度器挂起的核心原理

1. 调度器状态管理

FreeRTOS 通过 uxSchedulerSuspended 变量跟踪调度器状态:

  • 0:调度器正常运行,任务可根据优先级和时间片切换。
  • 非 0:调度器挂起,禁止任务切换(但中断仍可响应)。
2. 关键 API
// 挂起任务调度器(禁止任务切换)
void vTaskSuspendAll( void );// 恢复任务调度器,并检查是否需要进行上下文切换
BaseType_t xTaskResumeAll( void );

二、源码实现分析

1. vTaskSuspendAll () 的实现
// tasks.c
void vTaskSuspendAll( void )
{portDISABLE_INTERRUPTS();  // 关中断(防止竞争条件)// 增加调度器挂起计数uxSchedulerSuspended++;portENABLE_INTERRUPTS();  // 开中断
}
2. xTaskResumeAll () 的实现
// tasks.c
BaseType_t xTaskResumeAll( void )
{TCB_t *pxTCB;BaseType_t xAlreadyYielded = pdFALSE;portDISABLE_INTERRUPTS();  // 关中断// 减少调度器挂起计数uxSchedulerSuspended--;if( uxSchedulerSuspended == 0 ){/* 如果有任务需要切换,则标记上下文切换 */if( xYieldPending != pdFALSE ){/* 找出最高优先级的就绪任务 */pxTCB = pxCurrentTCB;taskSELECT_HIGHEST_PRIORITY_TASK();if( pxTCB != pxCurrentTCB ){/* 触发 PendSV 异常进行上下文切换 */portYIELD_WITHIN_API();xAlreadyYielded = pdTRUE;}else{xYieldPending = pdFALSE;}}}portENABLE_INTERRUPTS();  // 开中断return xAlreadyYielded;
}

三、调度器挂起与中断的关系

特性调度器挂起关中断
任务切换禁止禁止
中断响应允许(中断服务程序可执行)禁止(所有可屏蔽中断被屏蔽)
上下文切换延迟仅在调度器恢复后可能发生完全禁止,直到中断恢复
典型应用场景长时间操作(如文件系统)短时间原子操作(如寄存器配置)

四、应用场景

1. 保护长时间执行的操作
void perform_long_operation( void )
{// 挂起调度器(允许中断,但禁止任务切换)vTaskSuspendAll();// 长时间操作(如 Flash 读写、复杂计算)write_to_flash();// 恢复调度器(可能触发任务切换)xTaskResumeAll();
}
2. 批量更新共享资源
void update_multiple_resources( void )
{vTaskSuspendAll();// 更新多个共享资源(避免被其他任务打断)resource1 = value1;resource2 = value2;calculate_result();xTaskResumeAll();
}
3. 与临界区组合使用
void critical_operation( void )
{// 挂起调度器(防止任务切换)vTaskSuspendAll();// 进入临界区(防止中断干扰)taskENTER_CRITICAL();// 关键操作(如硬件初始化)init_peripheral();// 退出临界区taskEXIT_CRITICAL();// 恢复调度器xTaskResumeAll();
}

五、注意事项

  1. 调度器挂起时间应尽量短
    长时间挂起会导致高优先级任务无法执行,影响系统响应性。

  2. 禁止在调度器挂起期间调用阻塞 API
    如 vTaskDelay()xQueueReceive() 等,可能导致死锁。

  3. 中断中触发的任务切换会延迟执行
    若在调度器挂起期间,中断服务程序通过 xTaskNotifyFromISR() 等 API 请求任务切换,该切换会在调度器恢复后执行。

  4. 嵌套挂起需谨慎
    多次调用 vTaskSuspendAll() 需对应次数的 xTaskResumeAll(),否则会导致调度器状态异常。

六、总结

任务调度器挂起是 FreeRTOS 中一种强大的同步机制,适合在允许中断响应但禁止任务切换的场景中使用。与关中断相比,它提供了更细粒度的控制,既能保护关键代码,又能保持系统对紧急事件的响应能力。合理使用调度器挂起,是设计高效实时系统的关键。


文章转载自:

http://C20poi0a.wmgjq.cn
http://IDStEkUP.wmgjq.cn
http://cdCrusHb.wmgjq.cn
http://CSnM2qeb.wmgjq.cn
http://bJQqnDwa.wmgjq.cn
http://Z8zBi43v.wmgjq.cn
http://rdjcoDz8.wmgjq.cn
http://7wGIJEOg.wmgjq.cn
http://gMi0uK0W.wmgjq.cn
http://d6LofMln.wmgjq.cn
http://O5efqRdx.wmgjq.cn
http://b5h6bUyU.wmgjq.cn
http://unHtXiFM.wmgjq.cn
http://MysbQJb3.wmgjq.cn
http://unj4uHA2.wmgjq.cn
http://0NeI8iEk.wmgjq.cn
http://icBjsJlE.wmgjq.cn
http://Ck4cSrQL.wmgjq.cn
http://zweJBikq.wmgjq.cn
http://6kKEZDX0.wmgjq.cn
http://kEG33IBe.wmgjq.cn
http://D3k3mKGv.wmgjq.cn
http://yTi2DA5x.wmgjq.cn
http://mkqherYa.wmgjq.cn
http://8TEDjUWD.wmgjq.cn
http://K3i8JsCH.wmgjq.cn
http://u5N2aafS.wmgjq.cn
http://ukFSS7AP.wmgjq.cn
http://KmKKrqdu.wmgjq.cn
http://w185kUaw.wmgjq.cn
http://www.dtcms.com/wzjs/706285.html

相关文章:

  • 上海网站建设 浦东百度快照排名
  • 企业合作的响应式网站腾讯体育
  • 网站的侧边栏怎么做网站后台页面模板下载
  • 青岛高端网站制作公司项目管理网站开发
  • 郑州企业网站如何建设优秀产品创意设计案例
  • 北京市住房建设投资中心网站wordpress相关书籍
  • 网站开发文档范文wordpress如何转换为中文
  • 商城类型的网站怎么做wordpress官方主题下载地址
  • 网站后台的网址忘记了wordpress文章发布保存都不行
  • 网站建设总结论坛申请网站备案前置审批
  • 网站安全检测软件农产品公司网页设计
  • 嘉兴高端网站建设哈尔滨建设工程信息网查询系统
  • app 与网站赣州seo
  • 网站的领券商城怎么做神马seo服务
  • 海南做公司网站如何判断一个网站是恶意网站
  • html5在线制作网站模板深圳家装互联网网站
  • 公司网站优化软件大庆做网站的公司
  • 手机网站价格百度怎么推广网站
  • 网站建设方案哪家好 推荐网站备案现场
  • 福州品牌网站设计wordpress 重置
  • 网站一般用什么软件做怎么制作一个网页
  • 珠海市 网站建设网站开发费应该入什么科目
  • 宁波企业制作网站网页网站开发大概多少钱
  • 深圳做网站和视频宣传机构哪个平台可以接推广任务
  • 东莞市建设局网站首页个人网站的设计与实现摘要
  • 替别人做网站一个公司可以做多少网站
  • 网站背景如何做网站信息服务费怎么做分录
  • 给军方做网站套模板行不行潍坊网站做的好的公司
  • 用摄像头直播网站怎么做做网站销售的
  • 广东省住房和建设局网站wordpress aff