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

【Bluedroid】蓝牙HID DEVICE错误报告处理全流程源码解析

本文基于Android蓝牙协议栈代码,深入解析HID设备在接收非法指令(如无效的SET_REPORT)时的错误处理全流程,涵盖错误映射、协议封装、传输控制三大核心模块。重点剖析以下机制:

  1. HID协议规范错误码的动态转换策略

  2. 控制通道的优先级保障与拥塞回避机制

  3. 错误报告与正常数据报文的差异化处理逻辑

  4. 跨层级状态同步与连接恢复机制

一、流程概述

1.1 错误触发与协议栈传递

①应用层检测

上层应用通过report_error()触发错误处理,传入原始错误码(如HID_PAR_HANDSHAKE_RSP_ERR_INVALID_REP_ID)。

②协议栈封装

BTA_HdReportError():分配tBTA_HD_REPORT_ERR结构体,封装BTA_HD_API_REPORT_ERROR_EVT事件,通过bta_sys_sendmsg投递到系统队列。

③事件路由

bta_hd_hdl_event():将事件路由至状态机,触发bta_hd_report_error_act动作函数。

1.2 协议规范转换

①错误码标准化

HID_DevReportError():将原始错误码映射为HID协议定义的7种标准错误类型(如未知错误强制转换为HID_PAR_HANDSHAKE_RSP_ERR_UNKNOWN)。

②消息构建

构建HID_TRANS_HANDSHAKE类型报文,参数字段携带标准错误码,通过控制通道(CID=0x0040)发送。

1.3 物理层传输控制

①通道选择

强制使用控制通道(HID_CHANNEL_CTRL),确保错误报文优先传输。

②拥塞检测

hidd_conn_send_data()检查HID_CONN_FLAGS_CONGESTED标志,若通道拥塞直接返回HID_ERR_CONGESTED

③断连恢复

若检测到连接断开,丢弃待发错误报文(不同于数据报文的缓存机制),避免无效重传。

二、源码解析

report_error

packages/modules/Bluetooth/system/btif/src/btif_hd.cc
/********************************************************************************* Function         report_error** Description      Sends HANDSHAKE with error info for invalid SET_REPORT** Returns          bt_status_t*******************************************************************************/
static bt_status_t report_error(uint8_t error) {log::verbose("");// 状态检查:确保应用已注册if (!btif_hd_cb.app_registered) {log::warn("application not yet registered");return BT_STATUS_NOT_READY;}// 状态检查:确保蓝牙HID设备已启用if (btif_hd_cb.status != BTIF_HD_ENABLED) {log::warn("BT-HD not enabled, status={}", btif_hd_cb.status);return BT_STATUS_NOT_READY;}// 调用BTA层API发送错误报告BTA_HdReportError(error);return BT_STATUS_SUCCESS;
}

向 HID 主机发送错误报告,用于响应无效的 SET_REPORT 命令。

BTA_HdReportError

packages/modules/Bluetooth/system/bta/hd/bta_hd_api.cc
/********************************************************************************* Function         BTA_HdReportError** Description      This function is called when reporting error for set report** Returns          void*******************************************************************************/
void BTA_HdReportError(uint8_t error) {log::verbose("");tBTA_HD_REPORT_ERR* p_buf =(tBTA_HD_REPORT_ERR*)osi_malloc(sizeof(tBTA_HD_REPORT_ERR));p_buf->hdr.event = BTA_HD_API_REPORT_ERROR_EVT;p_buf->error = error;bta_sys_sendmsg(p_buf);
}

生成并发送包含错误信息的 HANDSHAKE 消息。

bta_hd_hdl_event(BTA_HD_API_REPORT_ERROR_EVT)

packages/modules/Bluetooth/system/bta/hd/bta_hd_api.cc
/********************************************************************************* Function         bta_hd_hdl_event** Description      HID device main event handling function.** Returns          void*******************************************************************************/
bool bta_hd_hdl_event(const BT_HDR_RIGID* p_msg) {log::verbose("p_msg->event={}", p_msg->event);switch (p_msg->event) {case BTA_HD_API_ENABLE_EVT:bta_hd_api_enable((tBTA_HD_DATA*)p_msg);break;case BTA_HD_API_DISABLE_EVT:if (bta_hd_cb.state == BTA_HD_CONN_ST) {log::warn("host connected, disconnect before disabling");// unregister (and disconnect)bta_hd_cb.disable_w4_close = TRUE;bta_hd_better_state_machine(BTA_HD_API_UNREGISTER_APP_EVT,(tBTA_HD_DATA*)p_msg);} else {bta_hd_api_disable();}break;default:bta_hd_better_state_machine(p_msg->event, (tBTA_HD_DATA*)p_msg);}return (TRUE);
}

负责接收和分发来自上层应用或下层协议栈的各种事件。作为 HID 设备状态机的入口点,根据不同的事件类型执行相应的处理逻辑。

bta_hd_better_state_machine

   ...case BTA_HD_CONN_ST:...case BTA_HD_API_REPORT_ERROR_EVT:bta_hd_report_error_act(p_data);break;...

bta_hd_report_error_act

packages/modules/Bluetooth/system/bta/hd/bta_hd_act.cc
/********************************************************************************* Function         bta_hd_report_error_act** Description** Returns          void*******************************************************************************/
void bta_hd_report_error_act(tBTA_HD_DATA* p_data) {tBTA_HD_REPORT_ERR* p_report = (tBTA_HD_REPORT_ERR*)p_data;tHID_STATUS ret;log::verbose("error = {}", p_report->error);ret = HID_DevReportError(p_report->error);if (ret != HID_SUCCESS) {log::warn("HID_DevReportError returned {}", ret);}
}

将错误信息封装为 HID 协议规范的错误报告并发送给主机设备。

HID_DevReportError

packages/modules/Bluetooth/system/stack/hid/hidd_api.cc
/********************************************************************************* Function         HID_DevReportError** Description      Reports error for Set Report via HANDSHAKE** Returns          tHID_STATUS*******************************************************************************/
tHID_STATUS HID_DevReportError(uint8_t error) {uint8_t handshake_param;log::verbose("error = {}", error);// 错误码映射switch (error) {case HID_PAR_HANDSHAKE_RSP_SUCCESS:case HID_PAR_HANDSHAKE_RSP_NOT_READY:case HID_PAR_HANDSHAKE_RSP_ERR_INVALID_REP_ID:case HID_PAR_HANDSHAKE_RSP_ERR_UNSUPPORTED_REQ:case HID_PAR_HANDSHAKE_RSP_ERR_INVALID_PARAM:case HID_PAR_HANDSHAKE_RSP_ERR_UNKNOWN:case HID_PAR_HANDSHAKE_RSP_ERR_FATAL:handshake_param = error;break;default:handshake_param = HID_PAR_HANDSHAKE_RSP_ERR_UNKNOWN;break;}// 构建 HANDSHAKE 消息return hidd_conn_send_data(0, HID_TRANS_HANDSHAKE, handshake_param, 0, 0,NULL);
}

负责将错误信息封装为 HID 协议规范的 HANDSHAKE 消息发送给主机设备。当设备接收到无效的 SET_REPORT 命令或其他错误条件时,通过此函数向主机反馈具体的错误类型。

对应 HID 协议规范中的标准错误类型:

错误码常量描述
HID_PAR_HANDSHAKE_RSP_SUCCESS0x00成功
HID_PAR_HANDSHAKE_RSP_NOT_READY0x01设备未准备好
HID_PAR_HANDSHAKE_RSP_ERR_INVALID_REP_ID0x02无效的报告 ID
HID_PAR_HANDSHAKE_RSP_ERR_UNSUPPORTED_REQ0x03不支持的请求
HID_PAR_HANDSHAKE_RSP_ERR_INVALID_PARAM0x04无效的参数
HID_PAR_HANDSHAKE_RSP_ERR_UNKNOWN0x14未知错误
HID_PAR_HANDSHAKE_RSP_ERR_FATAL0x15致命错误

hidd_conn_send_data

packages/modules/Bluetooth/system/stack/hid/hidd_conn.cc
/********************************************************************************* Function         hidd_conn_send_data** Description      Sends data to host** Returns          tHID_STATUS*******************************************************************************/
tHID_STATUS hidd_conn_send_data(uint8_t channel, uint8_t msg_type,uint8_t param, uint8_t data, uint16_t len,uint8_t* p_data) {BT_HDR* p_buf;uint8_t* p_out;uint16_t cid;uint16_t buf_size;log::verbose("channel({}), msg_type({}), len({})", channel, msg_type, len);tHID_CONN* p_hcon = &hd_cb.device.conn;// 1. 拥塞检查if (p_hcon->conn_flags & HID_CONN_FLAGS_CONGESTED) {log_counter_metrics(android::bluetooth::CodePathCounterKeyEnum::HIDD_ERR_CONGESTED_AT_FLAG_CHECK,1);return HID_ERR_CONGESTED;}// 2. 通道和缓冲区分配switch (msg_type) {case HID_TRANS_HANDSHAKE:case HID_TRANS_CONTROL:cid = p_hcon->ctrl_cid;buf_size = HID_CONTROL_BUF_SIZE;break;case HID_TRANS_DATA:if (channel == HID_CHANNEL_CTRL) {cid = p_hcon->ctrl_cid;buf_size = HID_CONTROL_BUF_SIZE;} else {cid = p_hcon->intr_cid;buf_size = HID_INTERRUPT_BUF_SIZE;}break;default:log_counter_metrics(android::bluetooth::CodePathCounterKeyEnum::HIDD_ERR_INVALID_PARAM,1);return (HID_ERR_INVALID_PARAM);}// 3. 消息封装p_buf = (BT_HDR*)osi_malloc(buf_size);if (p_buf == NULL) {log_counter_metrics(android::bluetooth::CodePathCounterKeyEnum::HIDD_ERR_NO_RESOURCES, 1);return (HID_ERR_NO_RESOURCES);}p_buf->offset = L2CAP_MIN_OFFSET;p_out = (uint8_t*)(p_buf + 1) + p_buf->offset;*p_out = HID_BUILD_HDR(msg_type, param); // 构建HID头部p_out++;p_buf->len = 1;  // start with header only// add report id prefix only if non-zero (which is reserved)if (msg_type == HID_TRANS_DATA && (data || param == HID_PAR_REP_TYPE_OTHER)) {*p_out = data;  // report_idp_out++;p_buf->len++;}if (len > 0 && p_data != NULL) {memcpy(p_out, p_data, len);  // 添加数据负载p_buf->len += len;}// 4. 连接状态检查// check if connectedif (hd_cb.device.state != HIDD_DEV_CONNECTED) {// for DATA on intr we hold transfer and try to reconnectif (msg_type == HID_TRANS_DATA && cid == p_hcon->intr_cid) {  // 保存待发送数据并尝试重新连接// drop previous data, we do not queue it for nowif (hd_cb.pending_data) {osi_free(hd_cb.pending_data);}hd_cb.pending_data = p_buf;if (hd_cb.device.conn.conn_state == HID_CONN_STATE_UNUSED) {hidd_conn_initiate();}return HID_SUCCESS;}log_counter_metrics(android::bluetooth::CodePathCounterKeyEnum::HIDD_ERR_NO_CONNECTION_AT_SEND_DATA,1);return HID_ERR_NO_CONNECTION;}log::verbose("report sent");// 5. 数据发送if (!L2CA_DataWrite(cid, p_buf)) {log_counter_metrics(android::bluetooth::CodePathCounterKeyEnum::HIDD_ERR_CONGESTED_AT_DATA_WRITE,1);return (HID_ERR_CONGESTED);}return (HID_SUCCESS);
}

负责将 HID 消息(如输入报告、控制命令、握手消息等)封装为 L2CAP 数据包并通过蓝牙物理层发送给主机设备。实现了数据发送的核心逻辑,包括通道选择、消息封装、连接状态检查和拥塞控制等功能。

三、关键机制解析

①错误码映射策略

  • 严格过滤:将用户输入的任意错误码映射到HID协议定义的7种类型,防止协议违规。

  • 安全兜底:未定义错误码统一转换为HID_PAR_HANDSHAKE_RSP_ERR_UNKNOWN,保障协议兼容性。

②控制通道保障机制

  • 独立缓冲区:使用固定512字节控制缓冲区(HID_CONTROL_BUF_SIZE),与数据通道隔离。

  • 优先级抢占:控制报文可中断正在传输的数据报文,确保错误及时反馈。

③差异化重传策略

报文类型重传机制缓存策略
数据报文自动重试3次保留最新数据
错误报文立即丢弃不缓存

④状态同步设计

  • 跨层状态机:通过bta_hd_cb.state同步应用层与协议栈状态,避免在禁用状态发送错误。

  • 原子化操作:错误处理全程关闭中断,通过osi_malloc保证内存操作的原子性。

四、时序图

五、总结

蓝牙 HID 协议中的错误报告机制是确保设备与主机之间可靠通信的重要组成部分。通过多层级的组件协作和标准化的消息格式,系统能够准确地将错误信息反馈给主机。核心代码实现采用了事件驱动架构、状态机模式和适配器模式,确保了系统的可扩展性和可维护性。

该流程体现了蓝牙HID设备错误处理的核心设计哲学:

  1. 协议合规性优先:通过严格的错误码映射和通道选择,确保符合HID 1.11规范。

  2. 故障弱化原则:在连接异常时主动丢弃错误报告,避免协议栈状态不一致。

  3. 实时性保障:控制通道的独立缓冲区设计和优先级机制,确保关键错误信息毫秒级送达。

  4. 状态驱动架构:通过事件-状态机模型实现跨层级状态同步,提升系统健壮性。


相关文章:

【Bluedroid】蓝牙HID DEVICE错误报告处理全流程源码解析

本文基于Android蓝牙协议栈代码,深入解析HID设备在接收非法指令(如无效的SET_REPORT)时的错误处理全流程,涵盖错误映射、协议封装、传输控制三大核心模块。重点剖析以下机制: HID协议规范错误码的动态转换策略 控制通…...

Day29 类的装饰器

类也有修饰器,他的逻辑类似:接收一个类,返回一个修改后的类。例如 添加新的方法或属性(如示例中的 log 方法)。修改原有方法(如替换 init 方法,添加日志)。甚至可以返回一个全新的类…...

学习黑客Active Directory 入门指南(二)

Active Directory 入门指南(二):深入逻辑结构与物理组件 🌳🏢 大家好!欢迎回到 “Active Directory 入门指南” 系列的第二篇。在上一篇中,我们初步认识了Active Directory,了解了其…...

为什么el-select组件在下拉选择后无法赋值

为什么el-select组件在下拉选择后无法赋值 https://blog.csdn.net/ZHENGCHUNJUN/article/details/127325558 这个链接解决了大模型无法解决的问题 大模型能够写基础且高级一些的代码,但是遇到再深入一些的问题,还是得问百度。对于我这种小白来说问题原因…...

FreeRTOS的学习记录(临界区保护,调度器挂起与恢复)

临界区 在 FreeRTOS 中,临界区(Critical Section) 是指程序中一段必须以原子方式执行的代码区域,在此区域内不允许发生任务切换或中断干扰,以保护共享资源或执行关键操作。FreeRTOS 提供了多种机制来实现临界区&#…...

给个人程序加上MCP翅膀

背景 最近MCP这个词真是到处都是,看起来特别高大上。我平时没事的时候也一直在关注这方面的技术,知道它是怎么一回事,也懂该怎么去实现。但可惜一直抽不出时间来自己动手搞一个MCP服务。网上关于MCP的教程一搜一大把,但基本上都是…...

2023年河南CCPC(ABCEFHK)

文章目录 2023河南CCPCA. 小水獭游河南(字符串)B. Art for Rest(数组切割)C. Toxel与随机数生成器(水)E. 矩阵游戏(dp)F. Art for Last(区间最小差分)H. Travel Begins(数学思维)K. 排列与质数(规律)总结 2023河南CCPC A. 小水獭…...

【 Redis | 实战篇 秒杀优化 】

目录 前言: 1.分布式锁 1.1.分布式锁的原理与方案 1.2.Redis的String结构实现分布式锁 1.3.锁误删问题 1.4.锁的原子性操作问题 1.5.Lua脚本解决原子性问题 1.6.基于String实现分布式锁存在的问题 1.7.Redisson分布式锁 2.秒杀优化 3.秒杀的异步优化 3.1…...

【Spring】核心机制:IOC与DI深度解析

目录 1.前言 2.正文 2.1三层架构 2.2Spring核心思想(IOC与AOP) 2.3两类注解:组件标识与配置 2.3.1五大类注解 2.3.1.1Controller 2.3.1.2Service 2.3.1.3Repository 2.3.1.4Configuration 2.3.1.5Component 2.3.2方法注解&#x…...

1-机器学习的基本概念

文章目录 一、机器学习的步骤Step1 - Function with unknownStep2 - Define Loss from Training DataStep3 - Optimization 二、机器学习的改进Q1 - 线性模型有一些缺点Q2 - 重新诠释机器学习的三步Q3 - 机器学习的扩展Q4 - 过拟合问题(Overfitting) 一、…...

ARM A64 STR指令

ARM A64 STR指令 1 STR (immediate)1.1 Post-index1.1.1 32-bit variant1.1.2 64-bit variant 1.2 Pre-index1.2.1 32-bit variant1.2.2 64-bit variant 1.3 Unsigned offset1.3.1 32-bit variant1.3.2 64-bit variant 1.4 Assembler symbols 2 STR (register)2.1 32-bit varia…...

虚幻引擎5-Unreal Engine笔记之`GameMode`、`关卡(Level)` 和 `关卡蓝图(Level Blueprint)`的关系

虚幻引擎5-Unreal Engine笔记之GameMode、关卡(Level) 和 关卡蓝图(Level Blueprint)的关系 code review! 文章目录 虚幻引擎5-Unreal Engine笔记之GameMode、关卡(Level) 和 关卡蓝图(Level B…...

软件工具:批量图片区域识别+重命名文件的方法,发票识别和区域选择方法参考,基于阿里云实现

基于阿里云的批量图片区域识别与重命名解决方案 图像识别重命名 应用场景 ​​企业档案管理​​:批量处理扫描的合同、文件等图片,根据合同编号、文件标题等关键信息重命名文件​​医疗影像处理​​:识别X光、CT等医学影像中的患者ID、检查日…...

.NET外挂系列:1. harmony 基本原理和骨架分析

一:背景 1. 讲故事 为什么要开这么一个系列,是因为他可以对 .NET SDK 中的方法进行外挂,这种技术对解决程序的一些疑难杂症特别有用,在.NET高级调试 领域下大显神威,在我的训练营里也是花了一些篇幅来说这个&#xf…...

深入理解位图(Bit - set):概念、实现与应用

目录 引言 一、位图概念 (一)基本原理 (二)适用场景 二、位图的实现(C 代码示例) 三、位图应用 1. 快速查找某个数据是否在一个集合中 2. 排序 去重 3. 求两个集合的交集、并集等 4. 操作系…...

React Flow 边事件处理实战:鼠标事件、键盘操作及连接规则设置(附完整代码)

本文为《React Agent:从零开始构建 AI 智能体》专栏系列文章。 专栏地址:https://blog.csdn.net/suiyingy/category_12933485.html。项目地址:https://gitee.com/fgai/react-agent(含完整代码示​例与实战源)。完整介绍…...

【计算机网络】第一章:计算机网络体系结构

本篇笔记课程来源:王道计算机考研 计算机网络 【计算机网络】第一章:计算机网络体系结构 一、计算机网络的概念1. 理论2. 计算机网络、互连网、互联网的区别 二、计算机网络的组成、功能1. 组成2. 功能 三、交换技术1. 电路交换2. 报文交换3. 分组交换4.…...

实战设计模式之状态模式

概述 作为一种行为设计模式,状态模式允许对象在其内部状态改变时,改变其行为。这种模式通过将状态逻辑从对象中分离出来,并封装到独立的状态类中来实现。每个状态类代表一种特定的状态,拥有自己的一套行为方法。当对象的状态发生变…...

[C++入门]类和对象中(2)日期计算器的实现

目录 一、运算符重载 1、格式 2、简单举例 2、前置,后置 3、日期生成器的实现 1、声明与定义 1、友元函数 2、print函数 3、运算符重载 4、GetMonthDay 5、,-,,-的实现 6、重载流操作符 2、实现 3、定义源码 一、运算…...

数据质量问题的形成与解决

在数字化时代,数据已成为企业和组织发展的核心资产,数据质量的高低直接影响着决策的准确性、业务的高效性以及系统的稳定性。然而,数据质量问题频发,严重阻碍了数据价值的充分发挥。 一、数据质量问题的成因分析 1.信息因素 元数…...

论文阅读(四):Agglomerative Transformer for Human-Object Interaction Detection

论文来源:ICCV(2023) 项目地址:https://github.com/six6607/AGER.git 1.研究背景 人机交互(HOI)检测需要同时定位人与物体对并识别其交互关系,核心挑战在于区分相似交互的细微视觉差异&#…...

【机器学习】工具入门:飞牛启动Dify Ollama Deepseek

很久没有更新文章了,最近正好需要研究一些机器学习的东西,打算研究一下 difyOllama 以下是基于FN 的dify本地化部署,当然这也可能是全网唯一的飞牛部署dify手册 部署 官方手册:https://docs.dify.ai/en/getting-started/install-self-hos…...

课外活动:再次理解页面实例化PO对象的魔法方法__getattr__

课外活动:再次理解页面实例化PO对象的魔法方法__getattr__ 一、动态属性访问机制解析 1.1 核心实现原理 class Page:def __getattr__(self, loc):"""魔法方法拦截未定义属性访问"""if loc not in self.locators.keys():raise Exce…...

面试题总结二

1.mybatis三个范式 第一范式:表中字段不能再分,每行数据都是唯一的第二范式:满足第一范式,非主键字段只依赖于主键第三范式:满足第二范式,非主键字段没有传递依赖 2.MySQL数据库引擎有哪些 InnoDB&#…...

代码随想录算法训练营第六十六天| 图论11—卡码网97. 小明逛公园,127. 骑士的攻击

继续补,又是两个新算法,继续进行勉强理解,也是训练营最后一天了,六十多天的刷题告一段落了! 97. 小明逛公园 97. 小明逛公园 感觉还是有点难理解原理 Floyd 算法对边的权值正负没有要求,都可以处理。核心…...

编程技能:字符串函数07,strncat

专栏导航 本节文章分别属于《Win32 学习笔记》和《MFC 学习笔记》两个专栏,故划分为两个专栏导航。读者可以自行选择前往哪个专栏。 (一)WIn32 专栏导航 上一篇:编程技能:字符串函数06,strcat 回到目录…...

[Java实战]Spring Boot整合RabbitMQ:实现异步通信与消息确认机制(二十七)

[Java实战]Spring Boot整合RabbitMQ:实现异步通信与消息确认机制(二十七) 摘要:本文通过完整案例演示Spring Boot与RabbitMQ的整合过程,深入讲解异步通信原理与消息可靠性保证机制。包含交换机类型选择、消息持久化配…...

数据库中关于查询选课问题的解法

前言 今天上午起来复习了老师上课讲的选课问题。我总结了三个解法以及一点注意事项。 选课问题介绍 简单来说就是查询某某同学没有选或者选了什么课。然后查询出该同学的姓名,学号,课程号,课程名之类的。 sql文件我上传了。大家可以尝试练…...

用 UniApp 开发 TilePuzzle:一个由 CodeBuddy 主动驱动的拼图小游戏

我正在参加CodeBuddy「首席试玩官」内容创作大赛,本文所使用的 CodeBuddy 免费下载链接:腾讯云代码助手 CodeBuddy - AI 时代的智能编程伙伴 起心动念:从一个小游戏想法开始 最近在使用 UniApp 做练手项目的时候,我萌生了一个小小…...

golang 安装gin包、创建路由基本总结

文章目录 一、安装gin包和热加载包二、路由简单场景总结 一、安装gin包和热加载包 首先终端新建一个main.go然后go mod init ‘项目名称’执行以下命令 安装gin包 go get -u github.com/gin-gonic/gin终端安装热加载包 go get github.com/pilu/fresh终端输入fresh 运行 &…...

组态王|组态王中如何添加西门子1200设备

哈喽,你好啊,我是雷工! 最近使用组态王采集设备数据,设备的控制器为西门子的1214CPU, 这里边实施边记录,以下为在组态王中添加西门子1200PLC的笔记。 1、新建 在组态王工程浏览器中选择【设备】→点击【新建】。 2、选择设备 和设备建立通讯要通过对应的设备驱动。 在…...

碎片笔记|PromptStealer复现要点(附Docker简单实用教程)

前言:本篇博客记录PromptStealer复现历程,主要分享环境配置过程中的一些经验。 论文信息:Prompt Stealing Attacks Against Text-to-Image Generation Models. USENIX, 2024. 开源代码:https://github.com/verazuo/prompt-stealin…...

Docker配置SRS服务器 ,ffmpeg使用rtmp协议推流+vlc拉流

目录 演示视频 前期配置 Docker配置 ffmpeg配置 vlc配置 下载并运行 SRS 服务 推拉流流程实现 演示视频 2025-05-18 21-48-01 前期配置 Docker配置 运行 SRS 建议使用 Docker 配置 Docker 请移步: 一篇就够!Windows上Docker Desktop安装 汉化完整指…...

c++学习之--- list

目录 ​编辑 一、list的定义: 二、list的模拟实现: 1、list的基本框架: 2、list的普通迭代器: 设计思想: 迭代器的一个特殊需求(c 对于重载->的一颗语法糖): 代码实现: 3、cons…...

【C++】set、map 容器的使用

文章目录 1. set 和 multiset 的使用1.1 set类的介绍1.2 set的构造和迭代器1.3 set 的增删查1.4 insert和迭代器调用示例1.5 find和erase使用示例1.6 multiset和set的差异 2. map 和 multimap 的使用2.1 map 类的介绍2.2 pair 类型介绍2.3 map 的构造和迭代器2.4 map 的增删查2…...

实习记录小程序|基于SSM+Vue的实习记录小程序设计与实现(源码+数据库+文档)

实习记录小程序 目录 基于SSM的习记录小程序设计与实现 一、前言 二、系统设计 三、系统功能设计 1、小程序端: 2、后台 四、数据库设计 五、核心代码 六、论文参考 七、最新计算机毕设选题推荐 八、源码获取: 博主介绍:✌️大厂码…...

Git从入门到精通

Git 是什么 Git 是一个分布式版本控制系统,主要用于跟踪和管理文件(尤其是代码)的变更。 Git的下载与安装 进入git官网下载界面,选择Windows系统。 点击选择Git for Windows/x64 Setup,进行安装。 注意: Git GUI 是Git提供的一个图形界面工…...

Binary Prediction with a Rainfall Dataset-(回归+特征工程+xgb)

Binary Prediction with a Rainfall Dataset 题意: 给你每天的天气信息,让你预测降雨量。 数据处理: 1.根据特征值构造天气降雨量的新特征值 2.根据时间构造月和季节特征 3.处理缺失值 建立模型: 1.建立lightgbm模型 2.建立…...

【C++】unordered_map与set的模拟实现

unordered系列map和set,与普通区别 用法几乎相同,键值唯一,区别unordered系列迭代器是单向的并且遍历出来不是有序的。unordered系列在数据规模大且无序的情况下性能更优 底层实现: map 和 set :基于平衡二叉树&…...

老旧设备升级利器:Modbus TCP转 Profinet让能效监控更智能

在工业自动化领域,ModbusTCP和Profinet是两种常见的通讯协议。Profinet是西门子公司推出的基于以太网的实时工业以太网标准,而Modbus则是由施耐德电气提出的全球首个真正开放的、应用于电子控制器上的现场总线协议。这两种协议各有各的优点,但…...

编译原理--期末复习

本文是我学习以下博主视频所作的笔记,写的不够清晰,建议大家直接去看这些博主的视频,他/她们讲得非常好: 基础知识概念: 1.【【编译原理】期末复习 零基础自学】,资料 2.【编译原理—混子速成期末保过】&…...

软件工程各种图总结

目录 1.数据流图 2.N-S盒图 3.程序流程图 4.UML图 UML用例图 UML状态图 UML时序图 5.E-R图 首先要先了解整个软件生命周期: 通常包含以下五个阶段:需求分析-》设计-》编码 -》测试-》运行和维护。 软件工程中应用到的图全部有:系统…...

Go 与 Gin 搭建简易 Postman:实现基础 HTTP 拨测的详细指南

Go 与 Gin 搭建简易 Postman:实现基础 HTTP 拨测的详细指南 文章目录 Go 与 Gin 搭建简易 Postman:实现基础 HTTP 拨测的详细指南项目简介代码结构各部分代码功能说明: 代码实现:main.go代码解释 handlers/probe.go代码解释 probe…...

层次原理图

层次原理图简介 层次原理图(Hierarchical Schematic)是一种常用于电子工程与系统设计的可视化工具,通过分层结构将复杂系统分解为多个可管理的子模块。它如同“设计蓝图”,以树状结构呈现整体与局部的关系:顶层展现系…...

嵌入式硬件篇---拓展板

文章目录 前言 前言 本文简单介绍了拓展板的原理以及使用。...

Redis的主从架构

主从模式 全量同步 首先主从同步过程第一步 会先比较replication id 判断是否是第一次同步假设为第一次同步 那么就会 启动bgsave异步生成RDB 同时fork子进程记录生成期间的新数据发送RDB给从节点 清空本地数据写入RDB 增量同步 对比ReplicationID不同因此选择增量同步在Rep…...

IIS入门指南:原理、部署与实战

引言:Web服务的基石 在Windows Server机房中,超过35%的企业级网站运行在IIS(Internet Information Services)之上。作为微软生态的核心Web服务器,IIS不仅支撑着ASP.NET应用的运行,更是Windows Server系统管…...

【上位机——WPF】布局控件

布局控件 常用布局控件Panel基类Grid(网格)UniformGrid(均匀分布)StackPanel(堆积面板)WrapPanel(换行面板)DockerPanel(停靠面板)Canvas(画布布局)Border(边框)GridSplitter(分割窗口)常用布局控件 Grid:网格,根据自定义行和列来设置控件的布局StackPanel:栈式面板,包含的…...

使用 C# 入门深度学习:线性代数详细讲解

在深度学习的领域中,线性代数是基础数学工具之一。无论是神经网络的训练过程,还是数据的预处理和特征提取,线性代数的知识都无处不在。掌握线性代数的核心概念,对于理解和实现深度学习算法至关重要。在本篇文章中,我们…...

操作系统之EXT文件系统

1.理解硬件 1.1磁盘、服务器、机柜、机房 机械磁盘是计算机中唯一的一个机械设备 磁盘--- 外设慢容量大,价格便宜 1.1.1光盘 1.1.2服务器 1.1.3机房 1.2磁盘的物理结构 1.3磁盘的存储结构 一个盘片又两个面 每个面都有一个磁头 磁头沿着盘面的半径移动 1.3.1…...