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 kotlin中 Channel 和 Flow 的区别和选择使用场景分析

《Androidkotlin中Channel和Flow的区别和选择使用场景分析》Kotlin协程中,Flow是冷数据流,按需触发,适合响应式数据处理;Channel是热数据流,持续发送,支持... 目录一、基本概念界定FlowChannel二、核心特性对比数据生产触发条件生产与消费的关系背压处理机制生命周期

Android ClassLoader加载机制详解

《AndroidClassLoader加载机制详解》Android的ClassLoader负责加载.dex文件,基于双亲委派模型,支持热修复和插件化,需注意类冲突、内存泄漏和兼容性问题,本文给大家介... 目录一、ClassLoader概述1.1 类加载的基本概念1.2 android与Java Class

Spring事务传播机制最佳实践

《Spring事务传播机制最佳实践》Spring的事务传播机制为我们提供了优雅的解决方案,本文将带您深入理解这一机制,掌握不同场景下的最佳实践,感兴趣的朋友一起看看吧... 目录1. 什么是事务传播行为2. Spring支持的七种事务传播行为2.1 REQUIRED(默认)2.2 SUPPORTS2

使用Python实现可恢复式多线程下载器

《使用Python实现可恢复式多线程下载器》在数字时代,大文件下载已成为日常操作,本文将手把手教你用Python打造专业级下载器,实现断点续传,多线程加速,速度限制等功能,感兴趣的小伙伴可以了解下... 目录一、智能续传:从崩溃边缘抢救进度二、多线程加速:榨干网络带宽三、速度控制:做网络的好邻居四、终端交互

MySQL中的锁机制详解之全局锁,表级锁,行级锁

《MySQL中的锁机制详解之全局锁,表级锁,行级锁》MySQL锁机制通过全局、表级、行级锁控制并发,保障数据一致性与隔离性,全局锁适用于全库备份,表级锁适合读多写少场景,行级锁(InnoDB)实现高并... 目录一、锁机制基础:从并发问题到锁分类1.1 并发访问的三大问题1.2 锁的核心作用1.3 锁粒度分

Redis的持久化之RDB和AOF机制详解

《Redis的持久化之RDB和AOF机制详解》:本文主要介绍Redis的持久化之RDB和AOF机制,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录概述RDB(Redis Database)核心原理触发方式手动触发自动触发AOF(Append-Only File)核

Android DataBinding 与 MVVM使用详解

《AndroidDataBinding与MVVM使用详解》本文介绍AndroidDataBinding库,其通过绑定UI组件与数据源实现自动更新,支持双向绑定和逻辑运算,减少模板代码,结合MV... 目录一、DataBinding 核心概念二、配置与基础使用1. 启用 DataBinding 2. 基础布局

Android ViewBinding使用流程

《AndroidViewBinding使用流程》AndroidViewBinding是Jetpack组件,替代findViewById,提供类型安全、空安全和编译时检查,代码简洁且性能优化,相比Da... 目录一、核心概念二、ViewBinding优点三、使用流程1. 启用 ViewBinding (模块级

基于 HTML5 Canvas 实现图片旋转与下载功能(完整代码展示)

《基于HTML5Canvas实现图片旋转与下载功能(完整代码展示)》本文将深入剖析一段基于HTML5Canvas的代码,该代码实现了图片的旋转(90度和180度)以及旋转后图片的下载... 目录一、引言二、html 结构分析三、css 样式分析四、JavaScript 功能实现一、引言在 Web 开发中,

springboot下载接口限速功能实现

《springboot下载接口限速功能实现》通过Redis统计并发数动态调整每个用户带宽,核心逻辑为每秒读取并发送限定数据量,防止单用户占用过多资源,确保整体下载均衡且高效,本文给大家介绍spring... 目录 一、整体目标 二、涉及的主要类/方法✅ 三、核心流程图解(简化) 四、关键代码详解1️⃣ 设置