android firmware下载机制

2024-05-15 22:38
文章标签 android 下载 机制 firmware

本文主要是介绍android firmware下载机制,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

加载固件的方法:

Linux 设备驱动的固件firmware加载

android 应用层

firmware文件要放入这三个目录之一:
/etc/firmware/
/vendor/firmware/
/firmware/image/

这里监听kernel发出的uevent(实际是监听netlink socket)。当收到uevent,根据名称在上面三个目录中寻找文件,找到文件后将文件传给kernel

android 中的ueventd是一个守护进程,主要作用是接收uevent来创建或删除/dev/xxx(设备节点),ueventd代码不多,下面我们直接针对代码分析

system/core/init/init.cpp

int main(int argc, char** argv) {if (!strcmp(basename(argv[0]), "ueventd")) {return ueventd_main(argc, argv);}

system/core/init/ueventd.cpp

int ueventd_main(int argc, char** argv) {/** init sets the umask to 077 for forked processes. We need to* create files with exact permissions, without modification by* the umask.*/umask(000);InitKernelLogging(argv);LOG(INFO) << "ueventd started!";selinux_callback cb;cb.func_log = selinux_klog_callback;selinux_set_callback(SELINUX_CB_LOG, cb);DeviceHandler device_handler = CreateDeviceHandler();UeventListener uevent_listener;if (access(COLDBOOT_DONE, F_OK) != 0) {ColdBoot cold_boot(uevent_listener, device_handler);cold_boot.Run();}// We use waitpid() in ColdBoot, so we can't ignore SIGCHLD until now.signal(SIGCHLD, SIG_IGN);// Reap and pending children that exited between the last call to waitpid() and setting SIG_IGN// for SIGCHLD above.while (waitpid(-1, nullptr, WNOHANG) > 0) {}uevent_listener.Poll([&device_handler](const Uevent& uevent) {HandleFirmwareEvent(uevent);device_handler.HandleDeviceEvent(uevent);return ListenerAction::kContinue;});return 0;
}

由HandleFirmwareEvent处理
system/core/init/firmware_handler.cpp

void HandleFirmwareEvent(const Uevent& uevent) {if (uevent.subsystem != "firmware" || uevent.action != "add") return;// Loading the firmware in a child means we can do that in parallel...auto pid = fork();if (pid == -1) {PLOG(ERROR) << "could not fork to process firmware event for " << uevent.firmware;}if (pid == 0) {Timer t;ProcessFirmwareEvent(uevent);LOG(INFO) << "loading " << uevent.path << " took " << t;_exit(EXIT_SUCCESS);}
}
static void ProcessFirmwareEvent(const Uevent& uevent) {int booting = IsBooting();LOG(INFO) << "firmware: loading '" << uevent.firmware << "' for '" << uevent.path << "'";std::string root = "/sys" + uevent.path;std::string loading = root + "/loading";std::string data = root + "/data";unique_fd loading_fd(open(loading.c_str(), O_WRONLY | O_CLOEXEC));if (loading_fd == -1) {PLOG(ERROR) << "couldn't open firmware loading fd for " << uevent.firmware;return;}unique_fd data_fd(open(data.c_str(), O_WRONLY | O_CLOEXEC));if (data_fd == -1) {PLOG(ERROR) << "couldn't open firmware data fd for " << uevent.firmware;return;}static const char* firmware_dirs[] = {"/etc/firmware/", "/vendor/firmware/","/firmware/image/"};try_loading_again:for (size_t i = 0; i < arraysize(firmware_dirs); i++) {std::string file = firmware_dirs[i] + uevent.firmware;unique_fd fw_fd(open(file.c_str(), O_RDONLY | O_CLOEXEC));struct stat sb;if (fw_fd != -1 && fstat(fw_fd, &sb) != -1) {LoadFirmware(uevent, root, fw_fd, sb.st_size, loading_fd, data_fd);return;}}if (booting) {// If we're not fully booted, we may be missing// filesystems needed for firmware, wait and retry.std::this_thread::sleep_for(100ms);booting = IsBooting();goto try_loading_again;}LOG(ERROR) << "firmware: could not find firmware for " << uevent.firmware;// Write "-1" as our response to the kernel's firmware request, since we have nothing for it.write(loading_fd, "-1", 2);
}

由代码可知从
static const char* firmware_dirs[] = {"/etc/firmware/", “/vendor/firmware/”,
“/firmware/image/”};

load到/sys/class/firmware/data中

request_firmware(const struct firmware **firmware_p第一个参数返回在内核空间的buf地址

一般需要厂商校验成功后再写到模块

这篇关于android firmware下载机制的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/993116

相关文章

Android协程高级用法大全

《Android协程高级用法大全》这篇文章给大家介绍Android协程高级用法大全,本文结合实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友跟随小编一起学习吧... 目录1️⃣ 协程作用域(CoroutineScope)与生命周期绑定Activity/Fragment 中手

基于Redis自动过期的流处理暂停机制

《基于Redis自动过期的流处理暂停机制》基于Redis自动过期的流处理暂停机制是一种高效、可靠且易于实现的解决方案,防止延时过大的数据影响实时处理自动恢复处理,以避免积压的数据影响实时性,下面就来详... 目录核心思路代码实现1. 初始化Redis连接和键前缀2. 接收数据时检查暂停状态3. 检测到延时过

Redis中哨兵机制和集群的区别及说明

《Redis中哨兵机制和集群的区别及说明》Redis哨兵通过主从复制实现高可用,适用于中小规模数据;集群采用分布式分片,支持动态扩展,适合大规模数据,哨兵管理简单但扩展性弱,集群性能更强但架构复杂,根... 目录一、架构设计与节点角色1. 哨兵机制(Sentinel)2. 集群(Cluster)二、数据分片

Python多线程实现大文件快速下载的代码实现

《Python多线程实现大文件快速下载的代码实现》在互联网时代,文件下载是日常操作之一,尤其是大文件,然而,网络条件不稳定或带宽有限时,下载速度会变得很慢,本文将介绍如何使用Python实现多线程下载... 目录引言一、多线程下载原理二、python实现多线程下载代码说明:三、实战案例四、注意事项五、总结引

Android 缓存日志Logcat导出与分析最佳实践

《Android缓存日志Logcat导出与分析最佳实践》本文全面介绍AndroidLogcat缓存日志的导出与分析方法,涵盖按进程、缓冲区类型及日志级别过滤,自动化工具使用,常见问题解决方案和最佳实... 目录android 缓存日志(Logcat)导出与分析全攻略为什么要导出缓存日志?按需过滤导出1. 按

深入理解go中interface机制

《深入理解go中interface机制》本文主要介绍了深入理解go中interface机制,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学... 目录前言interface使用类型判断总结前言go的interface是一组method的集合,不

C# async await 异步编程实现机制详解

《C#asyncawait异步编程实现机制详解》async/await是C#5.0引入的语法糖,它基于**状态机(StateMachine)**模式实现,将异步方法转换为编译器生成的状态机类,本... 目录一、async/await 异步编程实现机制1.1 核心概念1.2 编译器转换过程1.3 关键组件解析

Redis客户端连接机制的实现方案

《Redis客户端连接机制的实现方案》本文主要介绍了Redis客户端连接机制的实现方案,包括事件驱动模型、非阻塞I/O处理、连接池应用及配置优化,具有一定的参考价值,感兴趣的可以了解一下... 目录1. Redis连接模型概述2. 连接建立过程详解2.1 连php接初始化流程2.2 关键配置参数3. 最大连

Android Paging 分页加载库使用实践

《AndroidPaging分页加载库使用实践》AndroidPaging库是Jetpack组件的一部分,它提供了一套完整的解决方案来处理大型数据集的分页加载,本文将深入探讨Paging库... 目录前言一、Paging 库概述二、Paging 3 核心组件1. PagingSource2. Pager3.

Spring Security 单点登录与自动登录机制的实现原理

《SpringSecurity单点登录与自动登录机制的实现原理》本文探讨SpringSecurity实现单点登录(SSO)与自动登录机制,涵盖JWT跨系统认证、RememberMe持久化Token... 目录一、核心概念解析1.1 单点登录(SSO)1.2 自动登录(Remember Me)二、代码分析三、