百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术文章 > 正文

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

gudong366 2025-05-16 16:16 9 浏览

一、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来客制化新增路径。


(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。




参考文档:

https://www.kernel.org/doc/html/v4.13/driver-api/firmware/index.html



相关推荐

梦幻诛仙12职业1亿元宝版本架设教程(包含资源下载)

架设教程:1;上传脚本zx到bin文件夹,给权限chmod-R777/bin2;输入zx安装宝塔面板输入对应序号,中途输入Y继续3;安装好之后,会给个宝塔地址及账号密码,复制地址...

Java零基础入门,科普Java你应该了解什么

最近很多人问我想学Java但是一点基础没有,网上看了一堆学习线路图还是无从下手。今天耗时3小时整理了一套保姆级的Java入门教程,建议收藏按照线路图一点点学习。一、Java的概况Java是1995年6...

01.Java发展历史(java发展历史简要)

1.Java发展历史Java由SunMicrosystems公司(现为Oracle公司)的JamesGosling及其团队在1991年开发,最初命名为"Oak",后改名为"...

Ubuntu16.04.1安装Java8(ubuntu终端安装java)

上篇文章讲解了怎么在Windows下安装Java8《Windows10安装Java8》,这里讲解下怎么在Linux下安装Java。由于之前已经安装了Ubuntu16.04.1《VmwareWorkst...

性能测试能力提升-JVM GC监控和优化

一、背景接着上一篇的知识:性能测试能力提升-JVMGC原理,本篇文章,我们将主要介绍JVMGC监控和优化相关的知识:命令行方式监控GC图形化方式监控GC什么时候需要开始GC优化?GC优化的目的GC...

Spring Boot Jar 包秒变 Docker 镜像实现多环境部署

你是否在互联网大厂后端开发工作中,遇到过这样的困扰?当完成一个SpringBoot项目开发,准备将Jar包部署到不同环境时,却发现各个环境依赖不同、配置复杂,部署过程繁琐又容易出错,不仅耗费...

「JDK 11」关于 Java 模块系统,看这一篇就够了

继2014年3月Java8发布之后,时隔4年,2018年9月,Java11如期发布,其间间隔了Java9和Java10两个非LTS(LongTermSupp...

对Java学习的10条建议(对java的认识和理解)

不少Java的初学者一开始都是信心满满准备迎接挑战,但是经过一段时间的学习之后,多少都会碰到各种挫败,以下北风网就总结一些对于初学者非常有用的建议,希望能够给他们解决现实中的问题。Java编程的准备:...

JAVA入门教程-第1章 概述(java入门指南)

大道至简-JAVA入门教程在本教程中,你将学习Java语言的基础知识。Java基础内容涵盖:Java基础概念、Java词法结构、Java数组、Java流程控制、Java字符串、Java...

推荐一款Java音频视频编码器,很赞

Jave2是什么JAVE2(Java音频视频编码器)库是ffmpeg项目上的Java包装器。开发人员可以利用JAVE2将音频和视频文件从一种格式转码为另一种格式。在示例中,您可以将AVI文件转换为MP...

【JAVA教程】JAVA入门及开发环境安装

一、Java开发环境概述Java开发需要三个核心组件:JDK(JavaDevelopmentKit)-Java开发工具包,开发、编译、调试Java程序JRE(JavaRuntimeE...

Windows和Linux环境下的JDK安装教程

JavaDevelopmentKit(简称JDK),是Java开发的核心工具包,提供了Java应用程序的编译、运行和开发所需的各类工具和类库。它包括了JRE(JavaRuntimeEnviro...

记Tomcat优化方案(tomcat优化的几种方法)

Tomcat服务吞吐量评估方案问题:评估方案在一台8核16G的linux服务器上,使用tomcat容器部署服务。在正常情况下如何评估这个tomcat服务可处理的连接数,即服务的吞吐量,请在正常情况下考...

JVM GC诡异问题排查,k8s差点害死我……

前言本文将通过一个真实的生产环境案例,详细展示如何系统性地排查和解决JVM垃圾收集问题。这个案例涵盖了从问题发现、分析诊断到最终解决的完整过程,对于理解JVM调优实战具有重要的参考价值。系统背景我们的...

Thorium Reader - 功能强大的跨平台免费电子书阅读器

在日常阅读日益普及的今天,选择一款合适的电子书阅读器至关重要。ThoriumReader作为一款独具特色的阅读应用,正逐渐在众多同类产品中崭露头角,为用户带来了卓越的阅读体验。  跨平台的便捷性...