Linux固件子系统的实现机制简介

news2024/12/23 9:37:44

一、Linux固件子系统概述

固件是硬件设备自身执行的一段程序。固件一般存放在设备flash内。而出于成本和便利性的考虑,通常是先将硬件设备的运行程序打包为一个特定格式的固件文件,存储到终端系统内,通过终端系统给硬件设备进行升级。Linux内核开发过程中,开发人员调试外设驱动设备,比如触控,充电,线性马达,存储,WIFI设备等,同样存在需要更新固件的情况。在Linux系统中,设备驱动程序处于内核态,而固件文件处于用户态,因此需要一个安全稳定可靠的机制,用来确保设备驱动程序成功加载固件文件。为了解决设备驱动程序从内核态稳定加载用户态固件文件的问题,Linux系统提供了固件子系统。

二、Linux固件子系统实现机制

1. 流程简介:

Linux固件子系统基于sysfs 和uevent机制实现。

驱动程序调用固件系统函数接口申请固件之后,固件子系统使用固件编译内核的方式去获取固件;如果获取失败,就使用固件缓存的方式去获取固件;如果仍然获取失败,就使用默认路径内核直接查找的方式去获取固件。如果还是获取失败,就通过上报uevent消息给init进程。init进程则接收到uevent消息,过滤出subsystem类型为firmware的消息。init进程根据uevent消息内指向的固件信息去查找固件,通过sysfs提供的文件节点接口,把获取的固件内容从用户态写入内核态,从而使驱动程序,获取到固件文件的数据。

Linux固件系统提供了多种在不同场景下获取固件文件的方法。

1)直接编译到内核的方式;

2)固件缓存的方式;

3)直接根据内核指定路径的方式:

4)通过init进程来协助处理的方式;

2. 流程框图:

3. 主要函数接口:

主要函数接口:申请固件接口主要类型分为同步和异步。

通常申请固件的过程比较耗时,以及处理固件升级的过程比较耗时,因此可以采用异步函数接口实现,或者在驱动程序内先创建工作队列调用同步函数接口实现。其中:

  • 内核申请固件文件调用 request_firmware函数实现。
  • 内核获取固件文件后调用release_firmware释放相关的内存。

其中:

  • request_firmware_direct接口只在内核指定的路径内查找固件,不使用uevent机制来获取固件。
  • request_firmware_nowait接口是通过异步的工作队列去获取固件,可以起到不阻塞驱动probe时间的作用。

4. 实现过程:

(1)request_firmware实现流程:

request_firmware函数通过调用_request_firmware_prepare函数,设置不同的标志位,实现不同的差异功能。

a. _request_firmware_prepare函数:

在打开CONFIG_FW_LOADER宏开关基础上,首先通过调用fw_get_builtin_firmware函数的方式,判断固件文件是否编译到内核。

接着调用fw_lookup_and_allocate_buf函数,判断全局fw_cache结构内链表是否记录过当前请求firmware的name。如果不存在当前请求firmware的name,则动态分配对应的内存空间并且添加当前请求firmware的name到全局的fw_cache结构内的链表。

b. fw_get_filesystem_firmware函数:

主要是通过内核提供的默认路径去查找固件文件,调用kernel_read_file_from_path函数。如果没有查找到固件文件,则通过标志位FW_OPT_USERHELPER判断,是否启用USER_HELPER模式实现。

其中:

Firmware系统内默认路径如下:

默认路径可以通过kernel command line的方式来增加一个路径,通过module_param_string接口传递给变量path来客制化新增路径。

  资料直通车:Linux内核源码技术学习路线+视频教程内核源码

学习直通车:Linux内核源码内存调优文件系统进程管理设备驱动/网络协议栈

(2) USER_HELPER模式:

在内核打开CONFIG_FW_LOADER_USER_HELPER之后,才支持该功能。主要功能就是通过kernel上报uevent消息给到init进程,通过init进程获取固件信息写入底层sysfs节点。

a. fw_load_from_user_helper函数:

先调用fw_create_instance函数创建device设备,class文件和属性文件,以及分配firmware_priv结构体。

接着在/sys/class/firmware 下将创建一个目录,该目录使用设备名作为它的目录名。

该目录包含三个属性:

  • loading:

设置为 1:该属性由负责装载固件的用户空间设置1开始;

设置为 0:当装载过程完毕;

设置为 -1:将终止固件装载过程。

  • data:

用来接收固件数据,在设置完loading 后,用户空间进程把固件写入该属性。

  • device:

/sys/devices 下相应入口的符号链接。

  • timeout:

默认申请firmware通过uevent方式最大超时时间为60S,支持上层写入超时时间。b. _request_firmware_load函数:

首先先禁用uevent上报,通过调用device_add函数添加设备,触发调用firmware_uevent函数。其中,填充uevent上报的信息格式,包括固件的名称,超时时间,是否异步。

下一步则启用uevent上报功能,同时调用kobject_uevent函数,上报add动作类型给到上层ueventd。

接着调用fw_state_wait_timeout函数,在预设的超时时间内等待上层ueventd的处理。

若超时时间达到或者收到完成量唤醒,则释放之前申请的内存,释放device,class等内存信息。

(3)ueventd相关firmware处理流程

Ueventd是init进程内重要的模块,它主要处理selinux,dev设备创建,监听kernel上报uevent消息,firmware固件加载等内容。

a. FirmwareHandler处理流程:

FirmwareHandler内的HandleUevent方法主要是处理firmware固件加载和底层节点的交互流程。

首先先判断uevent消息的subsystem类型是firmware字段才进行处理,这个类型只有kernel内firmware模块才会上报。

HandleUevent主要是通过一个主线程创建不同的子线程,并行分别处理来自kernel的不同驱动的firmware请求。

b. ProcessFirmwareEvent函数:

首先是循环判断ueventd支持的路径内检索固件文件是否存在;若存在,则写入底层loading属性文件为1,同时拷贝获取的固件文件,写入到底层data文件。完成之后则写入底层loading属性文件为0。

至此,kernel就获取到了用户空间写入的固件文件信息。

其中:

ueventd 默认支持搜索固件的路径:

来自 ueventd.rc文件内指定的firmware_directory。

原文作者:内核工匠

 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/861822.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

self-attention(自注意力机制)

先举个有趣的例子理解 Q 、 K 、 V Q、K、V Q、K、V: 将我们要查询的内容,和商品列表进行相似度匹配,先拿出相似度更高的商品列表。 再根据以往的评价,计算出总分,按照分数进行排序。 self-attention d k \sqrt{d_k}…

ubuntu 安装 python

ubuntu 安装 python 初环境与设备查询是否安装安装python 本篇文章将介绍ubuntu 安装 python 初 希望能写一些简单的教程和案例分享给需要的人 环境与设备 系统:ubuntu 查询是否安装 因为系统也许会自带一个python,所以验证一下,如果自…

Linux mysql5.7开启 binlog

查看 mysql是否开启 binlog。 查看命令: show variables like %log_bin%; log_bin OFF 是关闭的状态。 编辑my.cnf配置文件 vim /etc/my.cnf 默认的配置文件内容: 增加下面内容 server_id 1 binlog_format ROW log-bin mysql_log_bin 重启mysq…

uniapp 微信小程序 订阅消息

第一步&#xff0c;需要先去小程序官方挑选一下订阅模板拿到模板id 订阅按钮在头部导航上&#xff0c;所以 <u-navbar :bgColor"bgColor"><view class"u-nav-slot" slot"left" click"goSubscribe"><image :src"g…

【Java】try|catch|throws 具体详解+应用

目录 tryCatch 基本介绍 使用细节 throws异常处理 基本介绍 ​ 使用细节 自定义异常 基本概念 步骤 throw和throws的区别 tryCatch 基本介绍 使用细节 throws异常处理 基本介绍 使用细节 自定义异常 基本概念 步骤 throw和throws的区别

常见的软件项目质量管理5种方法

产品质量的重要性不言而喻&#xff0c;为了确保项目质量&#xff0c;我们需要快速高效地找出影响产品质量的因素。如果不能及时高效洞察影响因素&#xff0c;无法及时修复项目缺陷和Bug&#xff0c;往往会对项目造成意想不到的后果和风险&#xff0c;如需求变更、重要开发节点延…

【雕爷学编程】Arduino动手做(201)---行空板开发环境之Jupyter

37款传感器与模块的提法&#xff0c;在网络上广泛流传&#xff0c;其实Arduino能够兼容的传感器模块肯定是不止37种的。鉴于本人手头积累了一些传感器和执行器模块&#xff0c;依照实践出真知&#xff08;一定要动手做&#xff09;的理念&#xff0c;以学习和交流为目的&#x…

数据库运维是什么意思?主要工作包含哪些?

还有不少小伙伴不知道数据库运维是什么意思&#xff1f;主要工作内容包含哪些&#xff1f;今天我们就一起来简单了解一下吧&#xff0c;仅供参考哦&#xff01; 数据库运维是什么意思&#xff1f; 数据库运维是指对数据库系统进行管理、监控和维护的过程&#xff0c;以确保数据…

CMU 15-445 -- Introduction to Distributed Databases - 19

CMU 15-445 -- Introduction to Distributed Databases - 19 引言System ArchitectureShared MemoryShared DiskShared Nothing Early Distributed Database SystemsDesign IssuesHomogeneous VS. Heterogeneous Database PartitioningNaive Table PartitioningHorizontal Part…

第一章 SpringBoot入门

1.SpringBoot简介 1.1.简介 Spring Boot来简化spring应用开发&#xff0c;约定大于配置去繁从简&#xff0c;just run就能创建一个独立的&#xff0c;产品级别的应用。 背景&#xff1a;J2EE笨重开发&#xff0c;繁多的配置、低下开发效率、复杂的部署流程、第三方技…

出现raise NotImplementedError报错

在学习《动手学深度学习》时&#xff0c;实现下面代码时&#xff0c;报出raise NotImplementedError错误。 import collections import torch from d2l import torch as d2l import math from torch import nnclass Seq2SeqEncoder(d2l.Encoder):def __init__(self,vocab_size,…

并发编程面试题2

并发编程面试题2 一、AQS高频问题&#xff1a; 1.1 AQS是什么&#xff1f; AQS就是一个抽象队列同步器&#xff0c;abstract queued sychronizer&#xff0c;本质就是一个抽象类。 AQS中有一个核心属性state&#xff0c;其次还有一个双向链表以及一个单项链表。 首先state…

NR sidelink(二) S-SSB

这篇看下NR sidelink S-SSB的内容,主要内容包括NR sidelink的同步原则,S-SSB的结构及相关序列,S-SSB的时频域位置,MasterInformationBlockSidelink IE解析,sidelink的同步过程,PSBCH payload及UE相关的能力IE。 NR sidelink的同步原则 参与sidelink通信之前,UE需要与覆…

VR全景智慧文旅,用科技助力旅游业振兴

引言&#xff1a; 近年来&#xff0c;科技的迅猛发展将我们带入一个全新的数字化时代&#xff0c;而虚拟现实&#xff08;Virtual Reality&#xff0c;简称VR&#xff09;技术则以其令人惊叹的全新方式&#xff0c;影响着各个领域。其中&#xff0c;旅游业作为人们探索世界、体…

AIGC接地项目机会网赚场景

变现场景 1、知识库 一个大公司有1000多份内部法规、合规的条文和规范&#xff0c;给员工普及这些知识很难&#xff0c;提前上课永远敌不过遇事抱佛脚。要是有一个能将1000条发文倒背如流的老司机&#xff0c;随时随地结合员工工作的实际场景&#xff0c;给出条文解释就好了。 …

多平台发布文章-项目总结

做个最近的AIGC内容创作技术要点的总结吧&#x1f63c; 流程图 时序图

研发工程师玩转Kubernetes——通过PV的节点亲和性影响Pod部署

在《研发工程师玩转Kubernetes——PVC通过storageClassName进行延迟绑定》一文中&#xff0c;我们利用Node亲和性&#xff0c;让Pod部署在节点ubuntud上。因为Pod使用的PVC可以部署在节点ubuntuc或者ubuntud上&#xff0c;而系统为了让Pod可以部署成功&#xff0c;则让PVC与Pod…

死锁的成因,和解决方案总结

何为死锁 死锁是多线程或并发程序中的一种情况&#xff0c;当多个线程因为竞争资源而相互等待&#xff0c;并且无法继续执行的情况。在死锁中&#xff0c;每个线程都在等待其他线程释放资源&#xff0c;从而导致所有线程都陷入无限等待状态&#xff0c;无法继续向前执行&#…

Python(七十六)字符串的驻留机制

❤️ 专栏简介&#xff1a;本专栏记录了我个人从零开始学习Python编程的过程。在这个专栏中&#xff0c;我将分享我在学习Python的过程中的学习笔记、学习路线以及各个知识点。 ☀️ 专栏适用人群 &#xff1a;本专栏适用于希望学习Python编程的初学者和有一定编程基础的人。无…