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

Zephyr RTOS 信号量 (Semaphore)

目录

概述

1 信号量类型及特性

1.1 信号量类型

1.2  信号量 vs 其他同步机制

2 核心 API 函数

2.1 信号量定义与初始化

2.2 信号量获取

2.3  信号量释放

2.4 信号量状态查询

2.5 重置信号量

3 典型应用场景

3.1 资源池管理(计数信号量)

3.2 生产者-消费者同步

3.3 互斥访问(临界区保护)

3.4 多线程协同工作

3.5 实战案例:多传感器数据采集系统

4 高级用法与技巧

4.1 中断服务程序 (ISR) 中的信号量

4.2 与超时结合使用

4.3 优先级继承(使用互斥信号量)

4.4 信号量链式同步

5 性能优化与最佳实践

5.1  避免优先级反转

5.2 减少信号量争用


概述

本文主要介绍Zephyr RTOS 信号量 (Semaphore) 的使用方法,信号量是 Zephyr RTOS 中最核心的同步机制之一,用于解决线程间同步、资源管理和互斥访问问题。Zephyr 的信号量机制提供了强大而灵活的线程同步能力。正确使用信号量可以构建出高效、响应迅速且资源利用率高的嵌入式系统。对于关键任务系统,其通过结合优先级继承和超时控制,确保系统在极端情况下的可靠性。

1 信号量类型及特性

1.1 信号量类型

类型初始值最大计数主要用途特点
计数信号量N (N>0)MAX (通常>1)资源池管理允许多个线程同时访问资源
二进制信号量0 或 11任务同步/互斥可替代互斥锁
互斥信号量11临界区保护支持优先级继承

1.2  信号量 vs 其他同步机制

机制最佳场景线程唤醒优先级继承中断安全
信号量资源计数/任务同步FIFO/优先级否(互斥除外)Give操作安全
互斥锁临界区保护优先级继承不安全
消息队列数据传输FIFOPut操作安全
事件多事件通知所有等待线程安全
条件变量复杂条件等待单个线程不安全

2 核心 API 函数

2.1 信号量定义与初始化

/* 静态定义 */
K_SEM_DEFINE(my_sem, initial_count, max_count);/* 动态初始化 */
struct k_sem my_sem;
k_sem_init(&my_sem, initial_count, max_count);

2.2 信号量获取

1)函数原型

int k_sem_take(struct k_sem *sem, k_timeout_t timeout);

2)参数

  • sem: 信号量指针

  • timeout: 超时设置(K_NO_WAIT, K_FOREVER, K_MSEC(ms))

 3)返回值

  • 0: 成功获取

  • -EAGAIN: 非阻塞模式下信号量不可用

  • -EINVAL: 无效参数

  • -ETIMEDOUT: 超时

 2.3  信号量释放

1)函数原型

void k_sem_give(struct k_sem *sem);

2)功能介绍

  • 在中断上下文安全使用

  • 如果线程在等待,唤醒优先级最高的线程

 2.4 信号量状态查询

函数原型:

// 获取当前信号量计数
unsigned int k_sem_count_get(struct k_sem *sem);// 检查信号量是否可用
int k_sem_takeable(struct k_sem *sem); // 1=可用, 0=不可用

2.5 重置信号量

函数原型:

void k_sem_reset(struct k_sem *sem); // 重置为初始值

3 典型应用场景

3.1 资源池管理(计数信号量)

#define MAX_CONNECTIONS 5
K_SEM_DEFINE(db_sem, MAX_CONNECTIONS, MAX_CONNECTIONS);void db_thread(void) {if (k_sem_take(&db_sem, K_MSEC(100)) {LOG_ERR("数据库连接超时");return;}// 访问数据库execute_query();k_sem_give(&db_sem); // 释放连接
}

3.2 生产者-消费者同步

K_SEM_DEFINE(data_ready, 0, 1); // 二进制信号量// 生产者
void sensor_thread(void) {while (1) {read_sensor_data();k_sem_give(&data_ready); // 通知消费者k_sleep(K_MSEC(10));}
}// 消费者
void process_thread(void) {while (1) {k_sem_take(&data_ready, K_FOREVER); // 等待数据process_data();}
}

3.3 互斥访问(临界区保护)

K_SEM_DEFINE(flash_mutex, 1, 1); // 二进制信号量void write_flash(void) {k_sem_take(&flash_mutex, K_FOREVER);// 临界区 - 安全访问Flashflash_write_data();k_sem_give(&flash_mutex);
}

3.4 多线程协同工作

K_SEM_DEFINE(step1_done, 0, 1);
K_SEM_DEFINE(step2_done, 0, 1);void thread_A(void) {perform_step1();k_sem_give(&step1_done); // 通知线程Bk_sem_take(&step2_done, K_FOREVER); // 等待线程Bperform_step3();
}void thread_B(void) {k_sem_take(&step1_done, K_FOREVER); // 等待线程Aperform_step2();k_sem_give(&step2_done); // 通知线程A
}

3.5 实战案例:多传感器数据采集系统

#include <zephyr/kernel.h>
#include <zephyr/drivers/sensor.h>K_SEM_DEFINE(temp_ready, 0, 1);
K_SEM_DEFINE(humid_ready, 0, 1);
K_SEM_DEFINE(data_mutex, 1, 1);struct sensor_data {float temperature;float humidity;
} current_data;// 温度传感器线程
void temp_thread(void) {const struct device *temp_sensor = DEVICE_DT_GET(DT_NODELABEL(temp0));while (1) {sensor_sample_fetch(temp_sensor);k_sem_take(&data_mutex, K_FOREVER);sensor_channel_get(temp_sensor, SENSOR_CHAN_AMBIENT_TEMP, &current_data.temperature);k_sem_give(&data_mutex);k_sem_give(&temp_ready);k_sleep(K_SECONDS(2));}
}// 湿度传感器线程
void humid_thread(void) {const struct device *humid_sensor = DEVICE_DT_GET(DT_NODELABEL(humid0));while (1) {sensor_sample_fetch(humid_sensor);k_sem_take(&data_mutex, K_FOREVER);sensor_channel_get(humid_sensor, SENSOR_CHAN_HUMIDITY, &current_data.humidity);k_sem_give(&data_mutex);k_sem_give(&humid_ready);k_sleep(K_SECONDS(3));}
}// 数据处理线程
void processing_thread(void) {while (1) {// 等待两种数据就绪k_sem_take(&temp_ready, K_FOREVER);k_sem_take(&humid_ready, K_FOREVER);k_sem_take(&data_mutex, K_FOREVER);float temp = current_data.temperature;float humid = current_data.humidity;k_sem_give(&data_mutex);// 计算露点float dew_point = calculate_dew_point(temp, humid);printk("温度: %.1fC, 湿度: %.1f%%, 露点: %.1fC\n", temp, humid, dew_point);}
}

4 高级用法与技巧

4.1 中断服务程序 (ISR) 中的信号量

K_SEM_DEFINE(adc_ready, 0, 1);// ADC转换完成中断
void adc_isr(const void *arg) {k_sem_give(&adc_ready); // 在ISR中安全释放信号量
}// 处理线程
void adc_thread(void) 
{while (1) {if (k_sem_take(&adc_ready, K_MSEC(500)) {process_adc_data();} else {LOG_WRN("ADC数据超时");}}
}

4.2 与超时结合使用

#define TIMEOUT_MS 200void critical_operation(void) {if (k_sem_take(&resource_sem, K_MSEC(TIMEOUT_MS))) {// 成功获取资源do_work();k_sem_give(&resource_sem);} else {// 超时处理handle_timeout();}
}

4.3 优先级继承(使用互斥信号量)

// 定义互斥信号量
K_MUTEX_DEFINE(high_pri_mutex);void high_pri_thread(void) {k_mutex_lock(&high_pri_mutex, K_FOREVER);// 临界区k_mutex_unlock(&high_pri_mutex);
}void low_pri_thread(void) {k_mutex_lock(&high_pri_mutex, K_FOREVER);// 执行期间会临时提升优先级k_mutex_unlock(&high_pri_mutex);
}

4.4 信号量链式同步

K_SEM_DEFINE(stage1, 0, 1);
K_SEM_DEFINE(stage2, 0, 1);
K_SEM_DEFINE(stage3, 0, 1);void pipeline_thread(void) {k_sem_take(&stage1, K_FOREVER);process_stage1();k_sem_give(&stage2);k_sem_take(&stage2, K_FOREVER);process_stage2();k_sem_give(&stage3);k_sem_take(&stage3, K_FOREVER);process_stage3();
}

5 性能优化与最佳实践

5.1  避免优先级反转

 

解决方案

  • 使用互斥信号量 (k_mutex) 替代二进制信号量

  • 限制临界区执行时间

  • 优先级继承机制

5.2 减少信号量争用

// 错误用法 - 频繁信号量操作
for (int i = 0; i < 100; i++) {k_sem_take(&sem, K_FOREVER);update_single_value(i);k_sem_give(&sem);
}// 优化用法 - 批量处理
k_sem_take(&sem, K_FOREVER);
for (int i = 0; i < 100; i++) {update_single_value(i);
}
k_sem_give(&sem);

http://www.dtcms.com/a/263702.html

相关文章:

  • 基于3D卷积神经网络与多模态信息融合的医学影像肿瘤分类与可视化分析
  • 商品中心—17.缓存与DB一致性的技术文档
  • Linux: network: 性能 pause
  • 【项目笔记】高并发内存池项目剖析(二)
  • Leetcode力扣解题记录--第49题(map)
  • 二型糖尿病居家管理小程序的设计与实现(消息订阅、websocket及时通讯、协同过滤算法)
  • OpenCV CUDA模块设备层-----“小于阈值设为零” 的图像处理函数thresh_to_zero_func()
  • 学习面向对象
  • 渗透测试(Penetration Testing)入门:如何发现服务器漏洞
  • 第2章,[标签 Win32] :兼容 ASCII 字符与宽字符的 Windows 函数调用
  • React:利用React.memo和useCallback缓存弹窗组件
  • 欢乐熊大话蓝牙知识24:LE Secure Connections 是 BLE 的安全升级术
  • HarmonyOS应用开发高级认证知识点梳理 (一) 布局与样式
  • 微信小程序使用秋云ucharts echarts
  • 利用springEvent,进行服务内部领域事件处理
  • 安卓接入百度定位和地图SDK
  • Go并发模式精要:掌握Goroutine与Channel的实战艺术
  • 开篇:GORM入门——Go语言的ORM王者
  • 物联网数据洪流下,TDengine 如何助 ThingLinks 实现 SaaS 平台毫秒级响应?
  • LabVIEW调用Excel宏实现数据可视化
  • 4D 毫米波雷达
  • 21、鸿蒙学习——使用App Linking实现应用间跳转
  • SQL Server 进阶:递归 CTE+CASE WHEN 实现复杂树形统计(第二课)
  • 【Python基础】12 闲谈分享:Python用于无人驾驶的未来
  • 借助飞算AI新手小白快速入门Java实操记录
  • 嵌入式编程-使用AI查找BUG的启发
  • AG32调试bug集合
  • [论文阅读] 人工智能 + 软件工程 | 从软件工程视角看大语言模型:挑战与未来之路
  • 基于 Vue + RuoYi 架构设计的商城Web/小程序实训课程
  • 企业级应用技术-ELK日志分析系统