从嵌入式Linux到嵌入式Android
最近开始投入Android的怀抱。说来惭愧,08年就听说这东西,当时也有同事投入去看,因为恶心Java,始终对这玩意无感,没想到现在不会这个嵌入式都快要没法搞了。为了不中年失业,所以只能回过头又来学。
首先还是说Android是基于Linux内核的,所以说骨子里还是linux,但是针对移动端,进行了深度优化。在这里结合GPT还有网上的信息,这里主要针对相比嵌入式Linux的差异,简单总结一下:
1 首先当然是前后端的分离。
有点类似Openwrt的发展。在之前,linux要开发图形程序是比较复杂的,QT,还有几个库,要么要收费,要么难以学习,最关键是和硬件平台绑定,换到新硬件要做移植。所以谷歌在操盘的时候,直接用Java做了前端GUI。当然,Java以前也有用J2SE做应用的传统。这样搞的好处也显而易见,前端和硬件分离,不管什么硬件,都不用改app代码了,虚拟机做适配就行了。IOS因为平台封闭,所以选了类C语言(具体机制空了再看看)。
所以首先的差异就是前端App要用Java开发,所以这个基本上绕不开。框架还是必须要学一下。
在前端App的层面,谷歌提供了Framework,还有大量的原生应用,比如电话,计算器,日历,email等。简化了很多开发。
=====================================
按照现在主流的分工,不管是Java还是Kotlin,应该都有专门的工程师。这部分对于嵌入式开发工程师我的理解是不用太懂,但是基本框架要懂,app是怎么怎么调用底层的,这个必须要知道。
试手一下CameraX(TODO)-CSDN博客
2 内核的变化(部分内容来自GPT)
Wakelocks: Android引入了"Wakelocks"机制,用于管理设备的唤醒状态。这是为了优化移动设备的电源管理,确保在需要时设备保持唤醒状态。就是俗称的唤醒锁,貌似这玩意主要是app层在用。
Low Memory Killer: Android引入了Low Memory Killer机制,用于监测并终止占用过多内存的应用程序进程,以维护系统的内存稳定性。这有助于提高系统的性能和响应能力。
Binder IPC(Inter-Process Communication): Android使用了自己的进程间通信机制,称为Binder。Binder允许Android系统中的不同组件进行通信,例如Activity、Service和BroadcastReceiver。据说这玩意类似Windows上的COM机制。
Ashmem: Android引入了Anonymous Shared Memory(Ashmem)机制,用于进程间共享匿名内存区域。这在Android中的共享内存和图形子系统中使用。
Logger: Android在内核中添加了用于日志记录的Logger模块,以支持Android的日志系统。
Android File System(FUSE): Android引入了FUSE(Filesystem in Userspace)用于文件系统的实现,允许用户空间程序运行文件系统代码而不需要修改内核。
安全性增强: Android对Linux内核进行了一些安全性的增强,包括SELinux(Security-Enhanced Linux)的集成,以提高系统的安全性。
调度策略: Android可能会使用不同的调度策略,以适应移动设备的性能和电源管理需求。
库的替换:好像glibc这些也都换了。
=====================================
大部分都是应用层。Binder是一个重头,因为涉及到接口。其他了解就行,真用到了再去看也可以。
Tee学习(TODO)-CSDN博客
3 驱动层的变化。(部分内容来自GPT)
Linux以前是标准的ioctl接口,Android改成了HAL接口,重要实现都切到了user space,用JNI封装。这些都要很了解。此外电源管理机制,还有Binder机制要看一下。
内核版本: Linux驱动接口: Linux驱动接口通常是为通用Linux内核设计的,支持各种硬件架构和设备类型。Linux内核在不同的发行版和版本中可能会有一些变化,但整体上是相对一致的。 Android驱动接口: Android使用了经过修改的Linux内核,因此Android驱动接口可能在某些方面与标准的Linux内核驱动接口略有不同。Android还可能引入一些额外的特定于移动设备的驱动需求。
电源管理: Linux驱动接口: Linux提供了通用的电源管理框架,驱动可以利用这些机制进行设备电源的管理。电源管理策略可能因硬件和内核配置而异。 Android驱动接口: Android对电源管理有其专有的需求,引入了Wakelocks等机制,以便更好地适应移动设备的电源管理和唤醒状态。
进程间通信(IPC): Linux驱动接口: 通常情况下,Linux驱动接口的设计并不直接涉及进程间通信,因为它主要关注设备与内核的交互。 Android驱动接口: Android引入了Binder机制,用于进程间通信。这对于Android中各种组件的通信非常重要,例如Activity、Service和BroadcastReceiver。
设备节点和HAL层: Linux驱动接口: 在标准Linux系统中,设备节点通常位于/dev目录下,用户空间可以通过这些设备节点与驱动进行通信。 Android驱动接口: Android引入了硬件抽象层(HAL),这是一个在用户空间和驱动之间的接口层,用于将Android系统与底层硬件驱动隔离开来。HAL层提供了标准接口,使得不同设备的驱动可以以相似的方式与Android系统进行交互。
调度策略: Linux驱动接口: Linux内核使用通用的进程调度策略,适用于各种设备和场景。 Android驱动接口: Android可能对调度策略进行定制,以适应移动设备的特殊需求,例如更好的响应性和电源管理。
驱动总览:


=====================================
这部分是重头,尤其是HAL层的差异,改的挺大的,必须要懂,而且要很懂。然后Android驱动层的一些实现,FrameBuffer(貌似现在换了),V4L2,ALSA,USB的gaeget框架,可能都要看看。当然,后面这部分也是Linux平台的玩法,区别不大。
Android的硬件接口HAL-CSDN博客
4 安卓运行环境增加的命令
am activity管理器,启动activity等。直接管理activity的原因可能还是为了调试方便。(猜测)
pm package管理器。这个没啥好多说的。
svc 服务管理器。和linux标准的service有点接近,但是可以管理的是Android的特定服务,比如蓝牙,wifi等等。
input 模拟输入,主要应该还是用在调试方面。(搞外挂利器?)
getprop/setprop 以前在路由器上,这一套要单独实现,就是TR069,SNMP之类,现在谷歌整体给你实现了。轻松很多。在嵌入式开发中,这个是大头。
settings 这个说的主要是设置android。但是和上面的有什么区别呢?还是看了下GPT,setting主要修改和用户界面相关的东西,主要是app领域。getprop/setprop则更广,包含一些系统底层属性,需要的权限也更高,有些要root权限。貌似在嵌入式开发中,主要还是后者。
安卓平台在本地也集成了一个数据库,有点类似SQLite。当然,这个就是爱用不用了。
getevent 获取本地事件。nandread 读取nand数据。
最后有别于一般用的busybox,安卓用的toolbox,常用命令大同小异。
=================================================
要点有两个,一个是通过getprop/setprop给adb调试接口,另一个是用getevent,inputs去做定位分析,也都不是太难。其他的知道怎么用就行了。
5 用户层差异
主要就是运行的文件系统,一些重要的守护进程,然后一些工具的了解。
守护进程列表:


安卓启动流程:

文件结构

重要服务的位置:

文件系统差异(来自GPT)
应用和用户数据: Android: Android 将应用和用户数据存储在 /data 分区中。每个应用程序都有自己的私有数据目录,其中包含其数据和设置。例如,应用程序的数据库、缓存和共享首选项通常存储在 /data/data/<package_name> 目录下。 嵌入式 Linux: 在一般的嵌入式 Linux 系统中,应用程序的数据和设置通常存储在 /usr、/var 或其他指定目录中。
系统文件和可执行文件: Android: Android 的系统文件和可执行文件通常存储在 /system 分区中。这包括 Android 操作系统的核心文件、系统应用和一些系统级别的设置。 嵌入式 Linux: 一般的嵌入式 Linux 系统的系统文件和可执行文件可能分散在不同的目录中,通常包括 /bin、/sbin、/lib 等。
可变数据和缓存: Android: 可变数据和缓存通常存储在 /cache 分区中。这包括一些临时文件和缓存,可能会在系统启动时被清理。 嵌入式 Linux: 一般的嵌入式 Linux 系统可能将临时文件和缓存存储在 /tmp 目录下。
外部存储: Android: 外部存储通常映射到 /sdcard,用于存储用户的媒体文件、下载内容等。 嵌入式 Linux: 嵌入式 Linux 系统也可以使用外部存储,但其挂载点和目录结构可能不同,具体取决于设备和系统定制。
硬件相关文件: Android: Android 中可能包含一些硬件相关的文件和节点,如 /dev 目录下的设备节点,用于与硬件交互。 嵌入式 Linux: 嵌入式 Linux 系统也有 /dev 目录,但硬件节点的命名和数量可能根据系统的硬件配置而变化。
=======================================
和第4点差不多,没什么难度,用两次熟悉就可以了,细节要用的时候知道怎么查就行了。
Android系统目录介绍_android 目录-CSDN博客
6 源码结构
看起来好像使用envsetup.sh,lunch还有make就够了。但是还是要深入了解一下。
这部分核心要点一个是如何新增设备,另一个是如何新增一个app。最后就是怎么在启动脚本里面加东西。
安卓的编译加速使用了CCache,我在之前公司曾经建议使用这玩意 ,不过后面被否了。。。
Gradle有点类似于makefile。还有一个要了解的就是Android.bp。感觉本质就是cmake那种东西。。。
============================================
这个也算次重点吧。主要是要懂编译环境,各个部分怎么生成的。要懂怎么增加一个设备,或者怎么增加一个App到编译环境。
CMake小结-CSDN博客
Android SDK学习(TODO)-CSDN博客
AOSP系列—阅读源码并熟悉AOSP目录结构_aospxref-CSDN博客
https://blog.51cto.com/u_7369281/1913427
7 Fastboot
也就是俗称的刷机模式。是一个Android特有的工具,在Linux上没有。Android通过特定手段进入该模型,上位机配合一个exe文件。实现操作固件的一些功能。
在我看来本质上就是对bootloader的一个封装,好像官方名也叫bootloader interface, 提供的功能也是uboot命令行的那些,分区,升级,刷固件,主备分区管理,系统变量的设置。常规嵌入式开发中,规划分区也是很重要的工作,貌似Android不能修改分区。
里面有Recovery模式和Fastboot模式,一个主要专注系统恢复,一个功能更全面。
=====================================================
如果用过uboot的话,看一下用法即可。
8 调试方法的学习
主要是ADB,这个比较简单,之前有写过。可能有一些细节,以后用到再说吧。
=====================================================
虽然用的很多,但是没什么难的。
ADB的使用-CSDN博客
9 平台特性的学习
主要是硬件平台的特性。
======================================================
这个也是重点,毕竟到什么山头唱什么歌,高通的,瑞芯微的,还是谁的,都要仔细学习。
就写这么多吧,还有的看到再写写。虽然说换汤不换药,本质核心还是编程能力。懂不懂这个汤可能就是会不会失业,运气好工资一个月差几十K也都有可能。所以还是学学吧。
参考资料:
Embedded Android
ChatGPT
相关文章:
从嵌入式Linux到嵌入式Android
最近开始投入Android的怀抱。说来惭愧,08年就听说这东西,当时也有同事投入去看,因为恶心Java,始终对这玩意无感,没想到现在不会这个嵌入式都快要没法搞了。为了不中年失业,所以只能回过头又来学。 首先还是…...
蓝桥ACM培训-实战1
前言: 今天老师没讲课,只让我们做了一下几道题目。 正文: Problem:A 小蓝与操作序列: #include<bits/stdc.h> using namespace std; stack<int> a; int main(){int n,flag1,ans;string cz;cin>>n;for(int i1;…...
波动数列(蓝桥杯)
问题描述: 观察如下数列: 1 3 0 2 -1 1 -2 … 这个数列中后一项总是比前一项增加 2 或者减少 3。 栋栋对这种数列很好奇,他想知道长度为 n nn 和为 s ss 而且后一项总是比前一项增加 a aa 或者减少 b bb 的整数数列可能有多少种呢?…...
第二篇【传奇开心果系列】Python的自动化办公库技术点案例示例:深度解读Pandas金融数据分析
传奇开心果博文系列 系列博文目录Python的自动化办公库技术点案例示例系列 博文目录前言一、Pandas 在金融数据分析中的常见用途和功能介绍二、金融数据清洗和准备示例代码三、金融数据索引和选择示例代码四、金融数据时间序列分析示例代码五、金融数据可视化示例代码六、金融数…...
Flink:Temporal Table Function(时态表函数)和 Temporal Join
博主历时三年精心创作的《大数据平台架构与原型实现:数据中台建设实战》一书现已由知名IT图书品牌电子工业出版社博文视点出版发行,点击《重磅推荐:建大数据平台太难了!给我发个工程原型吧!》了解图书详情,…...
Go语言中的时间控制:定时器技术详细指南
Go语言中的时间控制:定时器技术详细指南 引言定时器基础创建和使用time.Timer使用time.Ticker实现周期性任务定时器的内部机制小结 使用time.Timer实现简单的定时任务创建和启动定时器停止和重置定时器定时器的实际应用小结 利用time.Ticker处理重复的定时任务创建和…...
面试笔记系列六之redis+kafka+zookeeper基础知识点整理及常见面试题
目录 Redis redis持久化机制:RDB和AOF Redis 持久化 RDB的优点 RDB的缺点 AOF 优点 AOF 缺点 4.X版本的整合策略 Redis做分布式锁用什么命令? Redis做分布式锁死锁有哪些情况,如何解决? Redis如何做分布式锁?…...
Golang动态高效JSON解析技巧
JSON如今广泛用于配置和通信协议,但由于其定义的灵活性,很容易传递错误数据。本文介绍了如何使用mapstructure工具实现动态灵活的JSON数据解析,在牺牲一定性能的前提下,有效提升开发效率和容错能力。原文: Efficient JSON Data Ha…...
双重检验锁
双重检验锁:设计模式中的单例模式,细分为单例模式中的懒加载模式。 单例模式 单例模式:指的是一个类只有一个对象。最简单的实现方式是设一个枚举类,只有一个对象。缺点是当对象还没有被使用时,对象就已经创建存在了…...
【RISC-V 指令集】RISC-V DSP 扩展指令集介绍(一)
前言: 本笔记是基于对RISC-V DSP扩展指令集文档总结的,《P-ext-proposal.pdf》文档的关键内容如下: 主要介绍了RISC-V的P扩展指令集及其相关细节。 首先,对P扩展指令进行了概述,并列出了其与其他扩展重复的指令。 …...
RocketMQ - CentOS 7.x 安装单机版并测试
【安装前环境准备】检查是否安装好JDK(必要):java -version查看CPU信息: # cat /proc/cpuinfo # lscpu # getconf _NPROCESSORS_ONLN # cat /sys/devices/system/cpu/online # cat /proc/interrupts | egrep -i cpu查看内存信息: # free -hm …...
[JavaWeb玩耍日记]HTML+CSS+JS快速使用
目录 一.标签 二.指定css 三.css选择器 四.超链接 五.视频与排版 六.布局测试 七.布局居中 八.表格 九.表单 十.表单项 十一.JS引入与输出 十二.JS变量,循环,函数 十三.Array与字符串方法 十四.自定义对象与JSON 十五.BOM对象 十六.获取…...
如何使用ArcGIS Pro创建最低成本路径
虽然两点之间直线最短,但是在实际运用中,还需要考虑地形、植被和土地利用类型等多种因素,需要加权计算最低成本路径,这里为大家介绍一下计算方法,希望能对你有所帮助。 数据来源 教程所使用的数据是从水经微图中下载…...
Neoverse CSS N3:实现市场领先能效的最快途径
区分老的架构 从云到边缘,Arm Neoverse 提供无与伦比的性能、效率、设计灵活性和 TCO 优势,正在颠覆传统基础设施芯片。 我们看到云和超大规模服务运营商正在推动更高的计算密度。随着 128 核心 CPU 设计上市(Microsoft Cobalt、阿里巴巴 Y…...
JavaScript实现的计时器效果
之前做过电商网站倒计时的效果,今天在倒计时的基础上,把代码修改了一下,改为计时器效果,实现了以下功能: 1.点击“开始”后,按秒计时且“开始”文字变为“停止”; 2.点击“停止”,计…...
仿函数(Functor(c++))
定义 仿函数(Functor)是一个可以像函数那样被调用的类对象。这意味着它实现了operator(),使得类的对象可以像函数那样被调用。 仿函数的主要特点 它是一个类。它重载了operator()。可以通过创建该类的对象,并像函数那样调用该对…...
智能汽车加速车规级存储应用DS2431P+TR 汽车级EEPROM 存储器IC
DS2431PT&R是一款1024位1-Wire EEPROM芯片,由四页存储区组成,每页256位。数据先被写入一个8字节暂存器中,经校验后复制到EEPROM存储器。该器件的特点是,四页存储区相互独立,可以单独进行写保护或进入EPROM仿真模式…...
js json转换成字符串
js中JSON数据转换成字符串,可以使用JSON.stringify()方法。 var obj {name: "张三", age: 18, gender: "男"}; var jsonString JSON.stringify(obj); console.log(jsonString); // 输出 {"name":"张三","age"…...
Linux笔记--基本操作指令
一、查看日期与日历 1.date指令 显示日期 #用法1:dateCST: China Standard Time时区,中国标准时间 #用法2: date 指定格式 [常用格式]: "%Y-%m-%d"(%F): 2022-07-25 "%H:%M:%S"(%T): 14:53:44 "%F %T" #用法3: date -d "-1 da…...
论文阅读:基于超像素的图卷积语义分割(图结构数据)
#Superpixel-based Graph Convolutional Network for Semantic Segmentation github链接 引言 GNN模型根据节点特征周围的边来训练节点特征,并获得最终的节点嵌入。通过利用具有不同滤波核的二维卷积对来自附近节点的信息进行整合,给定超像素方法生成的…...
Prompt Tuning、P-Tuning、Prefix Tuning的区别
一、Prompt Tuning、P-Tuning、Prefix Tuning的区别 1. Prompt Tuning(提示调优) 核心思想:固定预训练模型参数,仅学习额外的连续提示向量(通常是嵌入层的一部分)。实现方式:在输入文本前添加可训练的连续向量(软提示),模型只更新这些提示参数。优势:参数量少(仅提…...
golang循环变量捕获问题
在 Go 语言中,当在循环中启动协程(goroutine)时,如果在协程闭包中直接引用循环变量,可能会遇到一个常见的陷阱 - 循环变量捕获问题。让我详细解释一下: 问题背景 看这个代码片段: fo…...
DockerHub与私有镜像仓库在容器化中的应用与管理
哈喽,大家好,我是左手python! Docker Hub的应用与管理 Docker Hub的基本概念与使用方法 Docker Hub是Docker官方提供的一个公共镜像仓库,用户可以在其中找到各种操作系统、软件和应用的镜像。开发者可以通过Docker Hub轻松获取所…...
【单片机期末】单片机系统设计
主要内容:系统状态机,系统时基,系统需求分析,系统构建,系统状态流图 一、题目要求 二、绘制系统状态流图 题目:根据上述描述绘制系统状态流图,注明状态转移条件及方向。 三、利用定时器产生时…...
Xen Server服务器释放磁盘空间
disk.sh #!/bin/bashcd /run/sr-mount/e54f0646-ae11-0457-b64f-eba4673b824c # 全部虚拟机物理磁盘文件存储 a$(ls -l | awk {print $NF} | cut -d. -f1) # 使用中的虚拟机物理磁盘文件 b$(xe vm-disk-list --multiple | grep uuid | awk {print $NF})printf "%s\n"…...
【Linux系统】Linux环境变量:系统配置的隐形指挥官
。# Linux系列 文章目录 前言一、环境变量的概念二、常见的环境变量三、环境变量特点及其相关指令3.1 环境变量的全局性3.2、环境变量的生命周期 四、环境变量的组织方式五、C语言对环境变量的操作5.1 设置环境变量:setenv5.2 删除环境变量:unsetenv5.3 遍历所有环境…...
mac:大模型系列测试
0 MAC 前几天经过学生优惠以及国补17K入手了mac studio,然后这两天亲自测试其模型行运用能力如何,是否支持微调、推理速度等能力。下面进入正文。 1 mac 与 unsloth 按照下面的进行安装以及测试,是可以跑通文章里面的代码。训练速度也是很快的。 注意…...
redis和redission的区别
Redis 和 Redisson 是两个密切相关但又本质不同的技术,它们扮演着完全不同的角色: Redis: 内存数据库/数据结构存储 本质: 它是一个开源的、高性能的、基于内存的 键值存储数据库。它也可以将数据持久化到磁盘。 核心功能: 提供丰…...
MFE(微前端) Module Federation:Webpack.config.js文件中每个属性的含义解释
以Module Federation 插件详为例,Webpack.config.js它可能的配置和含义如下: 前言 Module Federation 的Webpack.config.js核心配置包括: name filename(定义应用标识) remotes(引用远程模块࿰…...
FFmpeg avformat_open_input函数分析
函数内部的总体流程如下: avformat_open_input 精简后的代码如下: int avformat_open_input(AVFormatContext **ps, const char *filename,ff_const59 AVInputFormat *fmt, AVDictionary **options) {AVFormatContext *s *ps;int i, ret 0;AVDictio…...
